Spring-day01

从上图中我们可以得出对于Spring的学习主要包含四部分内容,分别是:

  • Spring的IOC/DI
  • Spring的AOP
  • AOP的具体应用,事务管理
  • IOC/DI的具体应用,整合Mybatis

1 Spring核心概念

使用对象时,在程序中不要主动使用new产生对象,转换为由外部提供对象,这种实现思就是Spring的一个核心概念

1.1 IOC

  1. 什么是IOC (Inversion of Control)控制反转呢?

    • 使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权由程序转移到外部,此思想称为控制反转。
      • 业务层要用数据层的类对象,以前是自己new
      • 现在自己不new了,交给别人[外部]来创建对象
      • 别人[外部]就反转控制了数据层对象的创建权
      • 这种思想就是控制反转
      • 别人[外部]指定是什么呢?继续往下学
  2. Spring和IOC之间的关系是什么呢?

    • Spring技术对IOC思想进行了实现

    • Spring提供了一个容器,称为IOC容器,用来充当IOC思想中的"外部"

    • IOC思想中的别人[外部]指的就是Spring的IOC容器

  3. IOC容器的作用以及内部存放的是什么?

    • IOC容器负责对象的创建、初始化等一系列工作,其中包含了数据层和业务层的类对象

    • 被创建或被管理的对象在IOC容器中统称为Bean

    • IOC容器中放的就是一个个的Bean对象

  4. 当IOC容器中创建好service和dao对象后,程序能正确执行么?

    • 不行,因为service运行需要依赖dao对象

    • IOC容器中虽然有service和dao对象

      • 但是service对象和dao对象没有任何关系
    • 需要把dao对象交给service,也就是说要绑定service和dao对象之间的关系,像这种在容器中建立对象与对象之间的绑定关系就要用到DI。

1.2 DI

  1. 什么是DI (Dependency Injection)依赖注入呢?

    • 在容器中建立bean与bean之间的依赖关系的整个过程,称为依赖注入
      • 业务层要用数据层的类对象,以前是自己new
      • 现在自己不new了,靠别人[外部其实指的就是IOC容器]来给注入进来
      • 这种思想就是依赖注入
  2. IOC容器中哪些bean之间要建立依赖关系呢?

    • 这个需要程序员根据业务需求提前建立好关系,如业务层需要依赖数据层,service就要和dao建立依赖关系

介绍完Spring的IOC和DI的概念后,我们会发现这两个概念的最终目标就是:充分解耦,具体实现靠:

  • 使用IOC容器管理bean(IOC)
  • 在IOC容器内将有依赖关系的bean进行关系绑定(DI)
  • 最终结果为:使用对象时不仅可以直接从IOC容器中获取,并且获取到的bean已经绑定了所有的依赖关系.

2. 入门案例

介绍完Spring的核心概念后,接下来我们得思考一个问题就是,Spring到底是如何来实现IOC和DI的,那接下来就通过一些简单的入门案例,来演示下具体实现过程:

2.1 IOC入门案例

2.1.1 入门案例思路分析

(1)Spring是使用容器来管理bean对象的,那么管什么?

  • 主要管理项目中所使用到的类对象,比如(Service和Dao)

(2)如何将被管理的对象告知IOC容器?

  • 使用配置文件

(3)被管理的对象交给IOC容器,要想从容器中获取对象,就先得思考如何获取到IOC容器?

  • Spring框架提供相应的接口

(4)IOC容器得到后,如何从容器中获取bean?

  • 调用Spring框架提供对应接口中的方法

(5)使用Spring导入哪些坐标?

  • 用别人的东西,就需要在pom.xml添加对应的依赖

2.1.2 入门案例代码实现

需求分析:将BookServiceImpl和BookDaoImpl交给Spring管理,并从容器中获取对应的bean对象进行方法调用。

1.创建Maven的java项目

2.pom.xml添加Spring的依赖jar包

3.创建BookService,BookServiceImpl,BookDao和BookDaoImpl四个类

4.resources下添加spring配置文件,并完成bean的配置

5.使用Spring提供的接口完成IOC容器的创建

6.从容器中获取对象进行方法调用

  1. 创建Maven项目

  2. 添加Spring的依赖jar包

    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.3.6</version>
    </dependency>
    
  3. 添加案例中需要的类

    public interface BookDao {
        public void save();
    }
    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println("book dao save ...");
        }
    }
    public interface BookService {
        public void save();
    }
    public class BookServiceImpl implements BookService {
        private BookDao bookDao = new BookDaoImpl();
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }
    
  4. 添加spring配置文件

    resources下添加spring配置文件applicationContext.xml,并完成bean的配置

  5. 在配置文件中完成bean的配置

    <!--1.导入spring的坐标spring-context,对应版本是5.2.10.RELEASE-->
    
    <!--2.配置bean-->
    <!--bean标签标示配置bean
    id属性标示给bean起名字
    class属性表示给bean定义类型-->
    <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl"/>
    <bean id="bookService" class="com.example.service.impl.BookServiceImpl">
    

    注意事项:bean定义时id属性在同一个上下文中(配置文件)不能重复

  6. 获取IOC容器

    使用Spring提供的接口完成IOC容器的创建,创建App类,编写main方法

    //3.获取IoC容器
    ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
    
  7. 从容器中获取对象进行方法调用

    public class App {
        public static void main(String[] args) {
            //获取IOC容器
    		ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml"); 
        BookDao bookDao = (BookDao) ctx.getBean("bookDao");
        bookDao.save();
        BookService bookService = (BookService) ctx.getBean("bookService");
        bookService.save();
        }
    }
    

**小结:**这样就可以通过Spring创建Java对象以及调用方法了,但是在BookServiceImpl的类中依然存在BookDaoImpl对象的new操作,它们之间的耦合度还是比较高,这块该如何解决,就需要用到下面的DI:依赖注入

2.2 DI入门案例

2.2.1 入门案例思路分析

(1)要想实现依赖注入,必须要基于IOC管理Bean

  • DI的入门案例要依赖于前面IOC的入门案例

(2)Service中使用new形式创建的Dao对象是否保留?

  • 需要删除掉,最终要使用IOC容器中的bean对象

(3)Service中需要的Dao对象如何进入到Service中?

  • 在Service中提供方法,让Spring的IOC容器可以通过该方法传入bean对象

(4)Service与Dao间的关系如何描述?

  • 使用配置文件

2.2.2 入门案例代码实现

需求:基于IOC入门案例,在BookServiceImpl类中删除new对象的方式,使用Spring的DI完成Dao层的注入

1.删除业务层中使用new的方式创建的dao对象

2.在业务层提供BookDao的setter方法

3.在配置文件中添加依赖注入的配置

4.运行程序调用方法

  1. 去除BookServiceImpl代码中的new BookDaoImpl()

    public class BookServiceImpl implements BookService {
        //删除业务层中使用new的方式创建的dao对象
        private BookDao bookDao;
    
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }
    
  2. 为属性提供setter方法

    public class BookServiceImpl implements BookService {
        //5.删除业务层中使用new的方式创建的dao对象
        private BookDao bookDao;
    
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
        //6.提供对应的set方法
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    }
    
  3. 修改配置完成注入

    <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl"/>
    
    <bean id="bookService" class="com.example.service.impl.BookServiceImpl">
      <!--7.配置server与dao的关系-->
      <!--property标签表示配置当前bean的属性
            name属性表示配置哪一个具体的属性,是class的set方法的变量
            ref属性表示需要注入哪一个bean class-->
      <property name="bookDao" ref="bookDao"/>
    </bean>
    
  4. 运行程序

    Main方法无需修改

3. IOC相关属性

通过前面两个案例,我们已经学习了bean如何定义配置DI如何定义配置以及容器对象如何获取的内容,接下来主要是把这三块内容展开进行详细的讲解,深入的学习下这三部分的内容,首先是bean基础配置。

3.1 bean基础配置

对于bean的配置中,主要会讲解bean基础配置,bean的别名配置,bean的作用范围配置(重点),这三部分内容:

3.1.1 bean的别名

对于bean的基础配置,在前面的案例中已经使用过:

<bean id="" class=""/>

其中,bean标签的功能、使用方式以及id和class属性的作用,我们通过一张图来描述下

image-20210729183500978

这其中需要大家重点掌握的是:bean标签的id和class属性的使用

思考:

  • class属性能不能写接口如BookDao的类全名呢?

答案肯定是不行,因为接口是没办法创建对象的。

  • 前面提过为bean设置id时,id必须唯一,但是如果由于命名习惯而产生了分歧后,该如何解决?

    首先来看下别名的配置说明:

  • 在配置文件中设置bean的别名

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
        <!--name:为bean指定别名,别名可以有多个,使用逗号,分号,空格进行分隔-->
        <bean id="bookService" name="service service4 bookEbi" class="com.example.service.impl.BookServiceImpl">
            <property name="bookDao" ref="bookDao"/>
        </bean>
        <!--scope:为bean设置作用范围,可选值为单例singloton,非单例prototype-->
        <bean id="bookDao" name="dao" class="com.example.dao.impl.BookDaoImpl" scope="prototype"/>
    </beans>
    
    
    

3.1.2 bean scope配置

关于bean的作用范围是bean属性配置的一个重点内容。

看到这个作用范围,我们就得思考bean的作用范围是来控制bean哪块内容的?

我们先来看下bean作用范围的配置属性:

验证IOC容器中对象是否为单例

验证思路

​ 同一个bean获取两次,将对象打印到控制台,看打印出的地址值是否一致。

具体实现

  • 创建一个AppForScope的类,在其main方法中来验证

    public class AppForScope {
        public static void main(String[] args) {
            ApplicationContext ctx = new 
                ClassPathXmlApplicationContext("applicationContext.xml");
    
            BookDao bookDao1 = (BookDao) ctx.getBean("bookDao");
            BookDao bookDao2 = (BookDao) ctx.getBean("bookDao");
            System.out.println(bookDao1);
            System.out.println(bookDao2);
        }
    }
    
  • 打印结果

    com.example.dao.impl.BookDaoImpl@e50a6f6
    com.example.dao.impl.BookDaoImpl@e50a6f6
    

    获取到结论后,问题就来了,那如果我想创建出来非单例的bean对象,该如何实现呢?

3.1.2.1 配置bean为非单例

在Spring配置文件中,配置scope属性来实现bean的非单例创建

在Spring的配置文件中,配置<bean>的scope属性

<!--scope:为bean设置作用范围,可选值为单例singloton,非单例prototype-->
<bean id="bookDao" name="dao" class="com.example.dao.impl.BookDaoImpl" scope="prototype"/>

scope使用后续思考:

介绍完scope属性以后,我们来思考几个问题:

  • 为什么bean默认为单例?
    • bean为单例的意思是在Spring的IOC容器中只会有该类的一个对象
    • bean对象只有一个就避免了对象的频繁创建与销毁,达到了bean对象的复用,性能高
  • bean在容器中是单例的,会不会产生线程安全问题?
    • 如果对象是有状态对象,即该对象有成员变量可以用来存储数据的,
    • 因为所有请求线程共用一个bean对象,所以会存在线程安全问题。
    • 如果对象是无状态对象,即该对象没有成员变量没有进行数据存储的,
    • 因方法中的局部变量在方法调用完成后会被销毁,所以不会存在线程安全问题。
  • 哪些bean对象适合交给容器进行管理?
    • 表现层对象
    • 业务层对象
    • 数据层对象
    • 工具对象
  • 哪些bean对象不适合交给容器进行管理?
    • 封装实例的域对象,因为会引发线程安全问题,所以不适合。

3.1.3 bean基础配置小结

关于bean的基础配置中,需要大家掌握以下属性:

3.2 bean实例化

对象已经能交给Spring的IOC容器来创建了,但是容器是如何来创建对象的呢?

就需要研究下bean的实例化过程,在这块内容中主要解决两部分内容,分别是

  • bean是如何创建的
  • 实例化bean的三种方式,构造方法,静态工厂实例工厂

在讲解这三种创建方式之前,我们需要先确认一件事:

bean本质上就是对象,对象在new的时候会使用构造方法完成,那创建bean也是使用构造方法完成的。

基于这个知识点出发,我们来验证spring中bean的三种创建方式。

3.2.1 构造方法实例化

  1. 准备需要被创建的类

准备一个BookDao和BookDaoImpl类

public interface BookDao {
    public void save();
}

public class BookDaoImpl implements BookDao {
    public void save() {
        System.out.println("book dao save ...");
    }

}
  1. 将类配置到Spring容器
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

	<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>

</beans>
  1. 编写运行程序
public class AppForInstanceBook {
    public static void main(String[] args) {
        ApplicationContext ctx = new 
            ClassPathXmlApplicationContext("applicationContext.xml");
        BookDao bookDao = (BookDao) ctx.getBean("bookDao");
        bookDao.save();

    }
}
  1. 类中提供构造函数测试

在BookDaoImpl类中添加一个无参构造函数私有化,并打印一句话,方便观察结果。

public class BookDaoImpl implements BookDao {
    private BookDaoImpl() {
        System.out.println("book dao constructor is running ....");
    }
    public void save() {
        System.out.println("book dao save ...");
    }

}

运行程序,如果控制台有打印构造函数中的输出,说明Spring容器在创建对象的时候也走的是构造函数。

并且能访问到类中的私有构造方法,显而易见Spring底层用的是反射。

3.2.2 静态工厂实例化

在讲这种方式之前,我们需要先回顾一个知识点是使用工厂来创建对象的方式:

(1)准备一个OrderDao和OrderDaoImpl类

public interface OrderDao {
    public void save();
}

public class OrderDaoImpl implements OrderDao {
    public void save() {
        System.out.println("order dao save ...");
    }
}

(2)创建一个工厂类OrderDaoFactory并提供一个静态方法

//静态工厂创建对象
public class OrderDaoFactory {
    public static OrderDao getOrderDao(){
        return new OrderDaoImpl();
    }
}

(3)编写AppForInstanceOrder运行类,在类中通过工厂获取对象

public class AppForInstanceOrder {
    public static void main(String[] args) {
        //通过静态工厂创建对象
        OrderDao orderDao = OrderDaoFactory.getOrderDao();
        orderDao.save();
    }
}

那么这个过程在Spring过程中如何实现呢?

  1. 在spring的配置文件application.properties中添加以下内容:

    <bean id="orderDao" class="com.itheima.factory.OrderDaoFactory" factory-method="getOrderDao"/>
    

    class:工厂类的类全名

    factory-mehod:具体工厂类中创建对象的方法名

  2. 在工厂的静态方法中,我们除了new对象还可以做其他的一些业务操作,这些操作必不可少

    public class OrderDaoFactory {
        public static OrderDao getOrderDao(){
            System.out.println("factory setup....");//模拟必要的业务操作
            return new OrderDaoImpl();
        }
    }
    
  3. 在AppForInstanceOrder运行类,使用从IOC容器中获取bean的方法进行运行测试

    public class AppForInstanceOrder {
        public static void main(String[] args) {
            ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
    
            OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");
    
            orderDao.save();
    
        }
    }
    

介绍完静态工厂实例化后,这种方式一般是用来兼容早期的一些老系统,所以了解为主**。**

3.2.3 实例工厂与FactoryBean

环境准备

(1)准备一个UserDao和UserDaoImpl类

public interface UserDao {
    public void save();
}

public class UserDaoImpl implements UserDao {

    public void save() {
        System.out.println("user dao save ...");
    }
}

(2)创建一个工厂类OrderDaoFactory并提供一个普通方法,注意此处和静态工厂的工厂类不一样的地方是方法不是静态方法

public class UserDaoFactory {
    public UserDao getUserDao(){
        return new UserDaoImpl();
    }
}

(3)编写AppForInstanceUser运行类,在类中通过工厂获取对象

public class AppForInstanceUser {
    public static void main(String[] args) {
        //创建实例工厂对象
        UserDaoFactory userDaoFactory = new UserDaoFactory();
        //通过实例工厂对象创建对象
        UserDao userDao = userDaoFactory.getUserDao();
        userDao.save();
}

对于上面这种实例工厂的方式如何交给Spring管理呢?

  1. 在spring的配置文件中添加以下内容:

    <bean id="userFactory" class="com.itheima.factory.UserDaoFactory"/>
    <bean id="userDao" factory-method="getUserDao" factory-bean="userFactory"/>
    

    实例化工厂运行的顺序是:

    • 创建实例化工厂对象,对应的是第一行配置

    • 调用对象中的方法来创建bean,对应的是第二行配置

      • factory-bean:工厂的实例对象
      • factory-method:工厂对象中的具体创建对象的方法名
      • factory-mehod:具体工厂类中创建对象的方法名
  2. 在AppForInstanceUser运行类,使用从IOC容器中获取bean的方法进行运行测试

    public class AppForInstanceUser {
        public static void main(String[] args) {
            ApplicationContext ctx = new 
                ClassPathXmlApplicationContext("applicationContext.xml");
            UserDao userDao = (UserDao) ctx.getBean("userDao");
            userDao.save();
        }
    }
    

实例工厂实例化的方式就已经介绍完了,配置的过程还是比较复杂,所以Spring为了简化这种配置方式就提供了一种叫FactoryBean的方式来简化开发。

3.2.3.1 FactoryBean的使用

具体的使用步骤为:

  1. 创建一个UserDaoFactoryBean的类,实现FactoryBean接口,确定要返回对象的泛型,并重写接口的方法

    public class UserDaoFactoryBean implements FactoryBean<UserDao> {
        //代替原始实例工厂中创建对象的方法
        public UserDao getObject() throws Exception {
            return new UserDaoImpl();
        }
        //返回所创建类的Class对象
        public Class<?> getObjectType() {
            return UserDao.class;
        }
    }
    

    查看源码会发现,FactoryBean接口其实会有三个方法,分别是:

    T getObject() throws Exception;
    
    Class<?> getObjectType();
    
    default boolean isSingleton() {
    		return true;
    }
    

    方法一:getObject(),被重写后,在方法中进行对象的创建并返回

    方法二:getObjectType(),被重写后,主要返回的是被创建类的Class对象

    方法三:isSingleton()没有被重写,因为它已经给了默认值,默认true,表示使用单例创建对象,重些return false,表示使用非单例。

  2. 在Spring的配置文件中进行配置

    <bean id="userDao" class="com.itheima.factory.UserDaoFactoryBean"/>
    
  3. AppForInstanceUser运行类不用做任何修改,直接运行

这种方式在Spring整合其他框架的时候会被用到,所以这种方式需要大家理解掌握

3.2.4 bean实例化小结

通过这一节的学习,需要掌握:

  1. bean是如何创建的呢?

    通过反射,使用构造函数创建对象

  2. Spring的IOC实例化对象的三种方式分别是:

    • 构造方法(常用)
    • 静态工厂(了解)
    • 实例工厂(了解)
      • FactoryBean(实用)

这些方式中,重点掌握构造方法FactoryBean即可。

需要注意的一点是,构造方法在类中默认会提供,但是如果重写了构造方法,默认的就会消失,在使用的过程中需要注意,如果需要重写构造方法,最好把默认的构造方法也重写下。

3.3 bean的生命周期

3.3.1 生命周期设置

bean的生命周期,具体的控制有两个阶段:

  • bean创建之后,想要添加内容,比如用来初始化需要用到资源
  • bean销毁之前,想要添加内容,比如用来释放用到的资源
  1. 添加初始化和销毁方法

    针对这两个阶段,我们在BooDaoImpl类中分别添加两个方法,方法名任意

    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println("book dao save ...");
        }
        //表示bean初始化对应的操作
        public void init(){
            System.out.println("init...");
        }
        //表示bean销毁前对应的操作
        public void destory(){
            System.out.println("destory...");
        }
    }
    
  2. 配置生命周期

    在配置文件添加配置,如下:

    <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl" init-method="init" destroy-method="destory"/>
    
  3. ApplicationContext中没有close方法,需要将ApplicationContext更换成ClassPathXmlApplicationContext

    ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
    
  4. 注册钩子关闭容器,或者调用ctx的close()方法

    ctx.registerShutdownHook();
    ctx.close();//二选一
    

    close和registerShutdownHook选哪个?

    相同点:这两种都能用来关闭容器

    不同点:close()是在调用的时候关闭,registerShutdownHook()是在JVM退出前调用关闭。

  5. 运行AppForLifeCycle类,打印结果如下

    init...
    book dao save ...
    destory...
    

分析上面的实现过程,会发现添加初始化和销毁方法,即需要编码也需要配置,实现起来步骤比较多也比较乱。

3.3.2 生命周期设置2

Spring提供了两个接口来完成生命周期的控制,好处是可以不用再进行配置init-methoddestroy-method

接下来在BookServiceImpl完成这两个接口的使用:

修改BookServiceImpl类,添加两个接口InitializingBeanDisposableBean并实现接口中的两个方法afterPropertiesSetdestroy

public class BookServiceImpl implements BookService, InitializingBean, DisposableBean {
    private BookDao bookDao;
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
    public void save() {
        System.out.println("book service save ...");
        bookDao.save(); 
    }
    public void destroy() throws Exception {
        System.out.println("service destroy");
    }
    public void afterPropertiesSet() throws Exception {
        System.out.println("service init");
    }
}

小细节

  • 对于InitializingBean接口中的afterPropertiesSet方法,翻译过来为属性设置之后

  • 对于BookServiceImpl来说,bookDao是它的一个属性

  • setBookDao方法是Spring的IOC容器为其注入属性的方法

  • setBookDao设置属性的方法会执行在afterPropertiesSet前

3.3.3 bean生命周期小结

(1)关于Spring中对bean生命周期控制提供了两种方式:

  • 在配置文件中的bean标签中添加init-methoddestroy-method属性
  • 类实现InitializingBeanDisposableBean接口,这种方式了解下即可。

(2)对于bean的生命周期控制在bean的整个生命周期中所处的位置如下:

  • 初始化容器
    • 1.创建对象(内存分配)
    • 2.执行构造方法
    • 3.执行属性注入(set操作)
    • 4.执行bean初始化方法
  • 使用bean
    • 1.执行业务操作
  • 关闭/销毁容器
    • 1.执行bean销毁方法

(3)关闭容器的两种方式:

  • ConfigurableApplicationContext是ApplicationContext的子类
    • close()方法
    • registerShutdownHook()方法

4. DI相关属性

前面我们已经完成了bean相关操作的讲解,接下来就进入第二个大的模块DI依赖注入,首先来介绍下Spring中有哪些注入方式?

我们先来思考

  • 向一个类中传递数据的方式有几种?
    • 普通方法(set方法)
    • 构造方法
  • 依赖注入描述了在容器中建立bean与bean之间的依赖关系的过程,如果bean运行需要的是数字或字符串呢?
    • 引用类型
    • 简单类型(基本数据类型与String)

Spring就是基于上面这些知识点,为我们提供了两种注入方式,分别是:

  • setter注入
    • 简单类型
    • 引用类型
  • 构造器注入
    • 简单类型
    • 引用类型

依赖注入的方式已经介绍完,接下来挨个学习下:

4.1 setter注入

4.1.1 注入引用数据类型

需求:在bookServiceImpl对象中注入bookDao

1.在BookServiceImpl中声明bookDao属性

2.为bookDao属性提供setter方法

3.在配置文件中使用property标签注入

  1. 对于setter方式注入引用类型的方式之前已经学习过,快速回顾下:
  • 在bean中定义引用类型属性,并提供可访问的set方法
public class BookServiceImpl implements BookService {
    private BookDao bookDao;
    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
}
  • 配置中使用property标签ref属性注入引用类型对象
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
	<property name="bookDao" ref="bookDao"/>
</bean>

<bean id="bookDao" class="com.itheima.dao.imipl.BookDaoImpl"/>

4.1.2 注入简单数据类型

给BookDaoImpl注入一些简单数据类型的数据

参考引用数据类型的注入,我们可以推出具体的步骤为:

1.在BookDaoImpl类中声明对应的简单数据类型的属性

2.为这些属性提供对应的setter方法

3.在applicationContext.xml中配置

思考:

引用类型使用的是<property name="" ref=""/>,简单数据类型还是使用ref么?

ref是指向Spring的IOC容器中的另一个bean对象的,对于简单数据类型,没有对应的bean对象,该如何配置?

  1. 声明属性并提供setter方法

    在BookDaoImpl类中声明对应的简单数据类型的属性,并提供对应的setter方法

    public class BookDaoImpl implements BookDao {
    
        private String databaseName;
        private int connectionNum;
        //setter注入需要提供要注入对象的set方法
        public void setConnectionNum(int connectionNum) {
            this.connectionNum = connectionNum;
        }
        //setter注入需要提供要注入对象的set方法
        public void setDatabaseName(String databaseName) {
            this.databaseName = databaseName;
        }
    
        public void save() {
            System.out.println("book dao save ..."+databaseName+","+connectionNum);
        }
    }
    
  2. 配置文件中进行注入配置

    在applicationContext.xml配置文件中使用property标签注入

    <!--注入简单类型-->
    <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl">
      <!--property标签:设置注入属性-->
      <!--name属性:设置注入的属性名,实际是set方法对应的名称-->
      <!--value属性:设置注入简单类型数据值-->
      <property name="connectionNum" value="100"/>
      <property name="databaseName" value="mysql"/>
    </bean>
    

    说明:

    value:后面跟的是简单数据类型,对于参数类型,Spring在注入的时候会自动转换,但是不能写成

    <property name="connectionNum" value="abc"/>
    

    这样的话,spring在将abc转换成int类型的时候就会报错。

    **注意:**两个property注入标签的顺序可以任意。

    对于setter注入方式的基本使用就已经介绍完了,

    • 对于引用数据类型使用的是<property name="" ref=""/>
    • 对于简单数据类型使用的是<property name="" value=""/>

4.2 构造器注入

构造器注入也就是构造方法注入

4.2.1 注入引用数据类型

需求:将BookServiceImpl类中的bookDao修改成使用构造器的方式注入。

1.将bookDao的setter方法删除掉

2.添加带有bookDao参数的构造方法

3.在applicationContext.xml中配置

  1. 删除setter方法并提供构造方法

    在BookServiceImpl类中将bookDao的setter方法删除掉,并添加带有bookDao参数的构造方法

    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
    
        public BookServiceImpl(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }
    
  2. 配置文件中进行配置构造方式注入

  3. 在applicationContext.xml中配置

    <bean id="userDao" class="com.example.dao.impl.UserDaoImpl"/>
    
    <bean id="bookService" class="com.example.service.impl.BookServiceImpl">
      <constructor-arg name="userDao" ref="userDao"/>
      <constructor-arg name="bookDao" ref="bookDao"/>
    </bean>
    

    说明:

    标签<constructor-arg>

    • name属性对应的值为构造函数中方法形参的参数名,必须要保持一致。

    • ref属性指向的是spring的IOC容器中其他bean对象。

4.2.2 注入简单数据类型

需求:在BookDaoImpl中,使用构造函数注入databaseName和connectionNum两个参数。

参考引用数据类型的注入,我们可以推出具体的步骤为:

1.提供一个包含这两个参数的构造方法

2.在applicationContext.xml中进行注入配置

  1. 添加多个简单属性并提供构造方法

    修改BookDaoImpl类,添加构造方法

    public class BookDaoImpl implements BookDao {
        private String databaseName;
        private int connectionNum;
    
        public BookDaoImpl(String databaseName, int connectionNum) {
            this.databaseName = databaseName;
            this.connectionNum = connectionNum;
        }
    
        public void save() {
            System.out.println("book dao save ..."+databaseName+","+connectionNum);
        }
    }
    
  2. 配置完成多个属性构造器注入

    在applicationContext.xml中进行注入配置

    <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl">
      根据构造方法参数名称注入
      <constructor-arg name="connectionNum" value="10"/>
      <constructor-arg name="databaseName" value="mysql"/>
    </bean>
    

    **说明:**这两个<contructor-arg>的配置顺序可以任意

上面已经完成了构造函数注入的基本使用,但是会存在一些问题:

  • 当构造函数中方法的参数名发生变化后,配置文件中的name属性也需要跟着变
  • 这两块存在紧耦合,具体该如何解决?

在解决这个问题之前,需要提前说明的是,这个参数名发生变化的情况并不多,所以上面的还是比较主流的配置方式,下面介绍的,大家都以了解为主。

方式一:删除name属性,添加type属性,按照类型注入

<bean id="bookDao" class="com.example.dao.impl.BookDaoImpl">
  根据构造方法参数类型注入
  <constructor-arg type="int" value="10"/>
  <constructor-arg type="java.lang.String" value="mysql"/>
</bean>
  • 这种方式可以解决构造函数形参名发生变化带来的耦合问题
  • 但是如果构造方法参数中有类型相同的参数,这种方式就不太好实现了

方式二:删除type属性,添加index属性,按照索引下标注入,下标从0开始

<!--解决参数类型重复问题,使用位置解决参数匹配-->
<bean id="bookDao" class="com.example.dao.impl.BookDaoImpl">
  <!--根据构造方法参数位置注入-->
  <constructor-arg index="0" value="mysql"/>
  <constructor-arg index="1" value="100"/>
</bean>
  • 这种方式可以解决参数类型重复问题
  • 但是如果构造方法参数顺序发生变化后,这种方式又带来了耦合问题

4.3 两种注入小结

介绍完两种参数的注入方式,具体我们该如何选择呢?

  1. 强制依赖使用构造器进行
    • 强制依赖指对象在创建的过程中必须要注入指定的参数
  2. 可选依赖使用setter注入进行,灵活性强
    • 可选依赖指对象在创建过程中注入的参数可有可无
  3. Spring框架倡导使用构造器,第三方框架内部大多数采用构造器注入的形式进行数据初始化,相对严谨
  4. 如果有必要可以两者同时使用,使用构造器注入完成强制依赖的注入,使用setter注入完成可选依赖的注入
  5. 实际开发过程中还要根据实际情况分析,如果受控对象没有提供setter方法就必须使用构造器注入
  6. 自己开发的模块推荐使用setter注入

这节中主要讲解的是Spring的依赖注入的实现方式:

  • setter注入

    • 简单数据类型

      <bean ...>
      	<property name="" value=""/>
      </bean>
      
    • 引用数据类型

      <bean ...>
      	<property name="" ref=""/>
      </bean>
      
  • 构造器注入

    • 简单数据类型

      <bean ...>
      	<constructor-arg name="" value=""/>
      </bean>
      
    • 引用数据类型

      <bean ...>
      	<constructor-arg name="" ref=""/>
      </bean>
      
  • 依赖注入的方式选择上

    • 建议使用setter注入
    • 第三方技术根据情况选择

5.4 自动配置

前面花了大量的时间把Spring的注入去学习了下,总结起来就一个字麻烦

问:有更简单方式么?

答:有,自动配置

什么是自动配置以及如何实现自动配置,就是接下来要学习的内容:

什么是依赖自动装配?

  • IoC容器根据bean所依赖的资源在容器中自动查找并注入到bean中的过程称为自动装配

自动装配方式有哪些?

  • 按类型(常用)
  • 按名称
  • 按构造方法
  • 不启用自动装配

5.4.1 自动装配的配置

自动装配只需要修改applicationContext.xml配置文件即可:

(1)将<property>标签删除

(2)在<bean>标签中添加autowire属性

首先来实现按照类型注入的配置

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean class="com.example.dao.impl.BookDaoImpl"/>
    <!--autowire属性:开启自动装配,通常使用按类型装配-->
    <bean id="bookService" class="com.example.service.impl.BookServiceImpl" autowire="byType"/>

</beans>

注意事项:

  • 需要注入属性的类中对应属性的setter方法不能省略
  • 被注入的对象必须要被Spring的IOC容器管理
  • 按照类型在Spring的IOC容器中如果找到多个对象,会报NoUniqueBeanDefinitionException

一个类型在IOC中有多个对象,还想要注入成功,这个时候就需要按照名称注入,配置方式为:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean name="bookDao" class="com.example.dao.impl.BookDaoImpl"/>
    <!--autowire属性:开启自动装配,通常使用按类型装配-->
    <bean id="bookService" class="com.example.service.impl.BookServiceImpl" autowire="byName"/>

</beans>

注意事项:

按照名称注入中的名称指的是什么?

其实是和对应的set方法有关,但是如果按照标准起名称,属性名和set对应的名是一致的

如果按照名称去找对应的bean对象,找不到则注入Null

5.4.2 自动装配小结

两种方式介绍完后,以后用的更多的是按照类型注入。

最后对于依赖注入,需要注意一些其他的配置特征:

  1. 自动装配用于引用类型依赖注入,不能对简单类型进行操作
  2. 使用按类型装配时(byType)必须保障容器中相同类型的bean唯一,推荐使用
  3. 使用按名称装配时(byName)必须保障容器中具有指定名称的bean,因变量名与配置耦合,不推荐使用
  4. 自动装配优先级低于setter注入与构造器注入,同时出现时自动装配配置失效

5.5 集合注入

前面我们已经能完成引入数据类型和简单数据类型的注入,但是还有一种数据类型集合,集合中既可以装简单数据类型也可以装引用数据类型,对于集合,在Spring中该如何注入呢?

先来回顾下,常见的集合类型有哪些?

  • 数组
  • List
  • Set
  • Map
  • Properties

针对不同的集合类型,该如何实现注入呢?

环境准备

  1. 创建一个Maven项目

  2. 项目中添加添加BookDao、BookDaoImpl类

    public interface BookDao {
        public void save();
    }
    
    public class BookDaoImpl implements BookDao {
        
    public class BookDaoImpl implements BookDao {
    
        private int[] array;
    
        private List<String> list;
    
        private Set<String> set;
    
        private Map<String,String> map;
    
        private Properties properties;
    
         public void save() {
            System.out.println("book dao save ...");
    
            System.out.println("遍历数组:" + Arrays.toString(array));
    
            System.out.println("遍历List" + list);
    
            System.out.println("遍历Set" + set);
    
            System.out.println("遍历Map" + map);
    
            System.out.println("遍历Properties" + properties);
        }
    	//setter....方法省略,自己使用工具生成
    }
    
  3. resources下提供spring的配置文件,applicationContext.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl"/>
    </beans>
    
  4. 在bookDao的bean标签中使用<property>进行注入

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
        <bean id="bookDao" class="com.example.dao.impl.BookDaoImpl">
            <!--数组注入-->
            <property name="array">
                <array>
                    <value>100</value>
                    <value>200</value>
                    <value>300</value>
                </array>
            </property>
            <!--list集合注入-->
            <property name="list">
                <list>
                    <value>itcast</value>
                    <value>itheima</value>
                    <value>boxuegu</value>
                    <value>chuanzhihui</value>
                </list>
            </property>
            <!--set集合注入-->
            <property name="set">
                <set>
                    <value>itcast</value>
                    <value>itheima</value>
                    <value>boxuegu</value>
                    <value>boxuegu</value>
                </set>
            </property>
            <!--map集合注入-->
            <property name="map">
                <map>
                    <entry key="country" value="china"/>
                    <entry key="province" value="henan"/>
                    <entry key="city" value="kaifeng"/>
                </map>
            </property>
            <!--Properties注入-->
            <property name="properties">
                <props>
                    <prop key="country">china</prop>
                    <prop key="province">henan</prop>
                    <prop key="city">kaifeng</prop>
                </props>
            </property>
        </bean>
    </beans>
    
    
  5. 配置完成后,运行下看结果:

    book dao save ...
    遍历数组:[100, 200, 300]
    遍历List[itcast, itheima, boxuegu, chuanzhihui]
    遍历Set[itcast, itheima, boxuegu]
    遍历Map{country=china, province=henan, city=kaifeng}
    遍历Properties{country=china, province=henan, city=kaifeng}
    
    

5.5.1注入数组类型数据

<property name="array">
    <array>
        <value>100</value>
        <value>200</value>
        <value>300</value>
    </array>
</property>

5.5.2 注入List类型数据

<property name="list">
    <list>
        <value>itcast</value>
        <value>itheima</value>
        <value>boxuegu</value>
        <value>chuanzhihui</value>
    </list>
</property>

5.5.3 注入Set类型数据

<property name="set">
    <set>
        <value>itcast</value>
        <value>itheima</value>
        <value>boxuegu</value>
        <value>boxuegu</value>
    </set>
</property>

5.5.4 注入Map类型数据

<property name="map">
    <map>
        <entry key="country" value="china"/>
        <entry key="province" value="henan"/>
        <entry key="city" value="kaifeng"/>
    </map>
</property>

5.5.5 注入Properties类型数据

<property name="properties">
    <props>
        <prop key="country">china</prop>
        <prop key="province">henan</prop>
        <prop key="city">kaifeng</prop>
    </props>
</property>

说明:

  • property标签表示setter方式注入,构造方式注入constructor-arg标签内部也可以写<array><list><set><map><props>标签
  • List的底层也是通过数组实现的,所以<list><array>标签是可以混用
  • 集合中要添加引用类型,只需要把<value>标签改成<ref>标签,这种方式用的比较少