Spring03——DI依赖注入(setter注入、构造器注入、依赖自动配置、集合注入)

DI相关内容

前面我们已经完成了bean相关操作的讲解,接下来就进入第二个大的模块DI依赖注入。

我们先来思考

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

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

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

11-setter注入

我们之前的案例就是使用setter注入引用类型,快速回顾下:

  1. 在bean中定义引用类型属性,并提供可访问的set方法
public class BookServiceImpl implements BookService {
    private BookDao bookDao;

    @Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
    }

    public void setBookDao(BookDao bookDao) {
        this.bookDao = bookDao;
    }
}
  1. 配置中使用property标签ref属性注入引用类型对象
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.blog.service.impl.BookServiceImpl">
    <property name="bookDao" ref="bookDao"></property>
</bean>
  1. 运行AppForDISet
public class AppForDISet {
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        BookService bookService = (BookService) ac.getBean("bookService");
        bookService.save();
    }
}

如果要注入多个引用类型(bean对象)该怎么做呢?
准备工作:
新建一个UserDao接口及其实现类

public interface UserDao {
    public void save();
}
public class UserDaoImpl implements UserDao {
    public void save() {
        System.out.println("user dao save ...");
    }
}

setter注入引用数据类型

需求:在bookServiceImpl对象中注入userDao

  1. 在BookServiceImpl中声明userDao属性
  2. 为userDao属性提供setter方法
  3. 在配置文件中使用property标签注入
  1. 在BookServiceImpl中声明userDao属性并提供setter方法
private BookDao bookDao;
// 声明属性
private UserDao userDao;
// 提供setter
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }
@Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
        // 执行userdao的方法
        userDao.save();
    }
  1. 在配置文件中注入配置
    在applicationContext.xml配置文件中将userDao定义成bean,使用property标签注入
<bean id="userDao" class="com.yolo.dao.impl.UserDaoImpl"/>

    <bean id="bookService" class="com.yolo.service.impl.BookServiceImpl">
        <property name="bookDao" ref="bookDao"/>
        <property name="userDao" ref="userDao"/>
    </bean>

在这里插入图片描述

  1. 运行程序,结果如下,userDao已经成功注入
    在这里插入图片描述

setter注入简单数据类型

需求:给BookDaoImpl注入一些简单数据类型的数据 参考引用数据类型的注入,我们可以推出具体的步骤为:

  1. 在BookDaoImpl类中声明对应的简单数据类型的属性
  2. 为这些属性提供对应的setter方法
  3. 在applicationContext.xml中配置

思考:
引用类型使用的是 <property name="" ref=""/>,简单数据类型还是使用 ref 吗?
答:
ref 是指向Spring的IOC容器中的另一个bean对象的,对于简单数据类型,没有对应的bean对象,该如何配置呢,使用value来配置<property name="" value=""/>

代码实现:

  1. 声明属性并提供setter方法
    这里举例就用String dataBaseName和int connectionCount这两个属性,同时在save()方法的输出语句中加上这两个属性
public class BookDaoImpl implements BookDao {

    private int connectionCount;
    private String dataBaseName;

    public void setConnectionCount(int connectionCount) {
        this.connectionCount = connectionCount;
    }

    public void setDataBaseName(String dataBaseName) {
        this.dataBaseName = dataBaseName;
    }

    @Override
    public void save() {
        System.out.println("book dao save..." + dataBaseName + "," + connectionCount );
    }
}
  1. 在配置文件中进行注入配置
    在applicationContext.xml配置文件中使用property标签注入
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl">
        <property name="connectionCount" value="10"/>
        <property name="dataBaseName" value="mysql"/>
    </bean>

value:后面跟的是简单数据类型,对于参数类型,Spring在注入的时候会自动转换,但是不能写一个错误的类型,
例如connectionCount是int类型,你却给他传一个abc,这样的话,spring在将abc转换成int类型的时候就会报错。

  • 运行程序,查看输出,两个简单数据类型也成功注入
    在这里插入图片描述

setter注入小结

对于setter注入方式

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

12-构造器注入

环境准备:

  1. BookDao及其实现类
public class BookDaoImpl implements BookDao {
    @Override
    public void save() {
        System.out.println("book dao save...");
    }
}
  1. BookService及其实现类
public class BookServiceImpl implements BookService {
    private BookDao bookDao;

    @Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
    }
}
  1. 配置文件
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>

    <bean id="bookService" class="com.yolo.service.impl.BookServiceImpl">
    
    </bean>
  1. 运行类
public class AppForDIConstructor {
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        BookService bookService = (BookService) ac.getBean("bookService");
        bookService.save();
    }
}

构造器注入引用数据类型

接下来,在上面这个环境中来完成构造器注入的学习:

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

  1. 将bookDao的setter方法删除掉并提供构造方法
  2. 添加带有bookDao参数的构造方法
  3. 在applicationContext.xml中配置
  1. 在BookServiceImpl类中将bookDao的setter方法删除掉,并添加带有bookDao参数的构造方法
public class BookServiceImpl implements BookService {
    private BookDao bookDao;

    public BookServiceImpl(BookDao bookDao) {
        this.bookDao = bookDao;
    }

    @Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
    }
}
  1. 配置文件中进行配置构造方式注入
    在applicationContext.xml中配置
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>

    <bean id="bookService" class="com.yolo.service.impl.BookServiceImpl">
        <constructor-arg name="bookDao1" ref="bookDao"/>
    </bean>

在这里插入图片描述
注意name中指代的是构造方法中的形参了

  1. 运行程序,结果说明bookDao已经成功注入。
    在这里插入图片描述

构造器注入多个引用数据类型

需求:在BookServiceImpl使用构造函数注入多个引用数据类型,比如userDao

  1. 声明userDao属性
  2. 生成一个带有bookDao和userDao参数的构造函数
  3. 在applicationContext.xml中配置注入
  1. 提供多个属性的构造函数
    在BookServiceImpl声明userDao并提供多个参数的构造函数,save方法中记得调用userDao.save()
public class BookServiceImpl implements BookService {
    private BookDao bookDao;
    private UserDao userDao;

    public BookServiceImpl(BookDao bookDao, UserDao userDao) {
        this.bookDao = bookDao;
        this.userDao = userDao;
    }

    @Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
        userDao.save();
    }
}
  1. 在配置文件中配置多参数注入
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>
    <bean id="userDao" class="com.yolo.dao.impl.UserDaoImpl"/>

    <bean id="bookService" class="com.yolo.service.impl.BookServiceImpl">
        <constructor-arg name="bookDao" ref="bookDao"/>
        <constructor-arg name="userDao" ref="userDao"/>
    </bean>
  1. 运行程序
    结果中出现了userDao的输出,说明userDao成功注入
    在这里插入图片描述

构造器注入多个简单数据类型

需求:在BookDaoImpl中,使用构造函数注入databaseName和connectionNum两个参数。
参考引用数据类型的注入,我们可以推出具体的步骤为:

  1. 提供一个包含这两个参数的构造方法
  2. 在applicationContext.xml中进行注入配置
  1. 添加多个简单属性并提供构造方法
    修改BookDaoImpl类,添加构造方法,同时在save()方法中输出这两个属性
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);
    }
}
  1. 配置完成多个属性构造器注入
<bean id="userDao" class="com.blog.dao.impl.UserDaoImpl"/>
    <bean id="bookDao" class="com.blog.dao.impl.BookDaoImpl">
        <constructor-arg name="databaseName" value="mysql"></constructor-arg>
        <constructor-arg name="connectionNum" value="10"></constructor-arg>
    </bean>
    <bean id="bookService" class="com.blog.service.impl.BookServiceImpl">
        <constructor-arg name="bookDao" ref="bookDao"/>
        <constructor-arg name="userDao" ref="userDao"/>
    </bean>
  1. 运行程序
    在这里插入图片描述

setter/构造器注入方式选择

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

  1. 强制依赖使用构造器注入,使用setter注入有概率不进行注入导致null对象出现
    • 强制依赖指bean对象在创建的过程中必须要注入指定的参数
    • setter方法实际使用中可执行可不执行,不像构造方法必须执行,使用setter注入如果不注入会导致null对象出现
  2. 可选依赖使用setter注入进行,灵活性强
    • 使用setter注入,不想注入的不进行配置即可
  3. Spring框架倡导使用构造器,第三方框架内部大多数采用构造器注入的形式进行数据初始化,相对严谨
  4. 如果有必要可以两者同时使用,使用构造器注入完成强制依赖的注入,使用setter注入完成可选依赖的注入(实际一般不同时用)
  5. 实际开发过程中还要根据实际情况分析,如果受控对象没有提供setter方法就必须使用构造器注入
    • 比如不是使用自己写的bean,而使用别人给的bean,这种情况就给什么用什么了
  6. 自己开发的模块推荐使用setter注入

13-依赖自动装配

前面花了大量的时间把Spring的注入去学习了下,总结起来就两个字麻烦。
问:麻烦在哪?
答:配置文件的编写配置上。
问:有更简单方式么?
答:有,自动配置
所以什么是自动配置以及如何实现自动配置,就是接下来要学习的内容

  1. 什么是依赖自动装配?
    IOC容器根据bean所依赖的资源在容器中自动查找(你要使用的资源)并注入到bean中的过程称为自动装配
  2. 自动装配的类型
    • 1.按类型(常用)
    • 2.按名称
    • 3.按构造方法
    • 4.不使用自动装配

环境准备:

  1. 修改BookDao、BookDaoImpl、BookService和BookServiceImpl类
public interface BookDao {
    public void save();
}
  1. 配置文件
<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>
    <bean id="bookService" class="com.yolo.service.impl.BookServiceImpl">
        <property name="bookDao" ref="bookDao"/>
    </bean>
  1. App运行类
public class App {
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        BookService bookService = (BookService) ac.getBean("bookService");
        bookService.save();
    }
}

完成自动装配的配置

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

  1. <property>标签删除
  2. <bean>标签中添加autowire属性

1.实现按照类型注入的配置

autowire="byType"

<bean id="bookDao" class="com.yolo.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.yolo.service.impl.BookServiceImpl" autowire="byType"/>
  • 运行程序,结果如下,说明已经成功注入了
    在这里插入图片描述

注意事项:

  1. 需要注入属性的类中对应属性的setter方法不能省略,不然会报错:NoSuchMethodException
  2. 被注入的对象必须要被Spring的IOC容器管理
  3. 按照类型在Spring的IOC容器中如果找到多个对象,会让程序找不到到底装配哪一个bean,就会报NoUniqueBeanDefinitionException
    在这里插入图片描述

2.实现按照名称注入的配置

autowire="byName"
当一个类型在IOC中有多个对象,还想要注入成功,这个时候就需要按照名称注入(提供的set方法的名称对应的id值的bean)

  • 修改配置方式如下
<!--这里就有两个同一类型的bean,但是id不一样-->
    <bean id="bookDao1" class="com.yolo.dao.impl.BookDaoImpl"/>
    <bean id="bookDao2" class="com.yolo.dao.impl.BookDaoImpl"/>

	<bean id="bookService" class="com.yolo.service.impl.BookServiceImpl" autowire="byName"/>
  • 同时修改BookServiceImpl类汇总的setBookDao方法,将其重命名为setBookDao1
public class BookServiceImpl implements BookService {
    private BookDao bookDao;

    public void setBookDao1(BookDao bookDao) {
        this.bookDao = bookDao;
    }

    @Override
    public void save() {
        System.out.println("book service save");
        bookDao.save();
    }
}
  • 运行程序,结果如下,说明已经成功注入了
    在这里插入图片描述

疑惑:为什么刚刚修改的是setBookDao的方法名,而不是将bookDao属性修改为bookDao1呢?按照名称注入中的名称指的是什么?
解惑:

  • 因为bookDao是private修饰的,外部类无法直接访问
  • 所以外部类只能通过属性的set方法进行访问
  • 对外部类来说,setBookDao方法名,去掉set后首字母小写是其属性名
    • 为什么是去掉set首字母小写?
    • 这个规则是set方法生成的默认规则,set方法的生成是把属性名首字母大写前面加set形成的方法名
  • 所以按照名称注入,其实是和对应的set方法有关,但是如果按照标准起名称,属性名和set对应的名是一致的

自动装配小结

  • 如果按照名称去找对应的bean对象,找不到则注入Null
  • 当某一个类型在IOC容器中有多个对象,按照名称注入只找其指定名称对应的bean对象,不会报错
  • 两种方式中,用的更多的是按照类型注入。
  • 对于依赖注入,需要注意一些其他的配置特征:
    1. 自动装配用于引用类型依赖注入,不能对简单类型进行操作
    2. 使用按类型装配时(byType)必须保障容器中相同类型的bean唯一,推荐使用
    3. 使用按名称装配时(byName)必须保障容器中具有指定名称的bean,因变量名与配置耦合,不推荐使用
    4. 自动装配优先级低于setter注入与构造器注入,同时出现时自动装配配置失效

14-集合注入

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

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

  • List
  • Set
  • Map
  • Properties

针对不同的集合类型,该如何实现注入呢?接着往下看
环境准备:

  1. 修改BookDaoImpl类
public interface BookDao {
    public void save();
}
  1. 修改配置文件
<?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.yolo.dao.impl.BookDaoImpl"/>
</beans>
  1. 修改App运行类
public class App {
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
        BookDao bookDao = (BookDao) ac.getBean("bookDao");
        bookDao.save();
    }
}

准备工作完毕,接下来,在上面这个环境中来完成集合注入的学习,下面所有的配置方式,都是在bookDao的bean标签中使用进行注入

  1. 在BookDaoImpl中创建属性并提供set方法
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 setArray(int[] array) {
        this.array = array;
    }

    public void setList(List<String> list) {
        this.list = list;
    }

    public void setSet(Set<String> set) {
        this.set = set;
    }

    public void setMap(Map<String, String> map) {
        this.map = map;
    }

    public void setProperties(Properties properties) {
        this.properties = properties;
    }

    public void save() {
        System.out.println("book dao save ...");
        System.out.println(Arrays.toString(array));
        System.out.println(list);
        System.out.println(set);
        System.out.println(map);
        System.out.println(properties);
    }
}

注入数组类型

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

注入List类型

<property name="list">
    <list>
        <value>张三</value>
        <value>ABC</value>
        <value>123</value>
    </list>
</property>

注入Set类型

<property name="set">
    <set>
        <value>100</value>
        <value>200</value>
        <value>ABC</value>
        <value>ABC</value>
    </set>
</property>

注入Map类型

<property name="map">
    <map>
        <entry key="探路者" value="马文"/>
        <entry key="次元游记兵" value="恶灵"/>
        <entry key="易位窃贼" value="罗芭"/>
    </map>
</property>

注入Properties类型

<property name="properties">
    <props>
        <prop key="暴雷">沃尔特·菲茨罗伊</prop>
        <prop key="寻血猎犬">布洛特·亨德尔</prop>
        <prop key="命脉">阿杰·切</prop>
    </props>
</property>
  1. 配置完成之后,运行查看结果
    在这里插入图片描述

说明:

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值