Spring中有哪些注入方式?
我们先来思考
向一个类中传递数据的方式有几种?
- 普通方法(set方法)
- 构造方法
依赖注入描述了在容器中建立bean与bean之间的依赖关系的过程,如果bean运行需要的是数字或
字符串呢?
- 引用类型
- 简单类型(基本数据类型与String)
Spring就是基于上面这些知识点,为我们提供了两种注入方式,分别是:
setter注入
- 简单类型
- 引用类型
构造器注入
- 简单类型
- 引用类型
依赖注入的方式已经介绍完,接下来挨个学习下:
setter注入
1. 对于setter方式注入引用类型的方式之前已经学习过,快速回顾下:
在bean中定义引用类型属性,并提供可访问的set方法
1 public class BookServiceImpl implements BookService {
2 private BookDao bookDao;
3 public void setBookDao(BookDao bookDao) {
4 this.bookDao = bookDao;
5 }
6 }
配置中使用property标签ref属性注入引用类型对象
1 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
2 <property name="bookDao" ref="bookDao"/>
3 </bean>
4
5 <bean id="bookDao" class="com.itheima.dao.imipl.BookDaoImpl"/>
环境准备
为了更好的学习下面内容,我们依旧准备一个新环境:
- 创建一个Maven项目
- pom.xml添加依赖
- resources下添加spring的配置文件
这些步骤和前面的都一致,大家可以快速的拷贝即可,最终项目的结构如下:
(1)项目中添加BookDao、BookDaoImpl、UserDao、UserDaoImpl、BookService和
BookServiceImpl类
1 public interface BookDao {
2 public void save();
3 }
4
5 public class BookDaoImpl implements BookDao {
6 public void save() {
7 System.out.println("book dao save ...");
8 }
9 }
10 public interface UserDao {
11 public void save();
12 }
13 public class UserDaoImpl implements UserDao {
14 public void save() {
15 System.out.println("user dao save ...");
16 }
17 }
18
19 public interface BookService {
20 public void save();
21 }
22
23 public class BookServiceImpl implements BookService{
24 private BookDao bookDao;
25
26 public void setBookDao(BookDao bookDao) {
27 this.bookDao = bookDao;
28 }
29
30 public void save() {
31 System.out.println("book service save ...");
32 bookDao.save();
33 }
34 }
(2)resources下提供spring的配置文件
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
8 <property name="bookDao" ref="bookDao"/>
9 </bean>
10 </beans>
(3)编写AppForDISet运行类,加载Spring的IOC容器,并从中获取对应的bean对象
1 public class AppForDISet {
2 public static void main( String[] args ) {
3 ApplicationContext ctx = new
ClassPathXmlApplicationContext("applicationContext.xml");
4 BookService bookService = (BookService) ctx.getBean("bookService");
5 bookService.save();
6 }
7 }
接下来,在上面这个环境中来完成setter注入的学习:
注入引用数据类型
需求:在bookServiceImpl对象中注入userDao
1.在BookServiceImpl中声明userDao属性
2.为userDao属性提供setter方法
3.在配置文件中使用property标签注入
步骤1:声明属性并提供setter方法
在BookServiceImpl中声明userDao属性,并提供setter方法
1 public class BookServiceImpl implements BookService{
2 private BookDao bookDao;
3 private UserDao userDao;
4
5 public void setUserDao(UserDao userDao) {
6 this.userDao = userDao;
7 }
8 public void setBookDao(BookDao bookDao) {
9 this.bookDao = bookDao;
10 }
11
12 public void save() {
13 System.out.println("book service save ...");
14 bookDao.save();
15 userDao.save();
16 }
17 }
步骤2:配置文件中进行注入配置
在applicationContext.xml配置文件中使用property标签注入
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
8 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
9 <property name="bookDao" ref="bookDao"/>
10 <property name="userDao" ref="userDao"/>
11 </bean>
12 </beans>
步骤3:运行程序
运行AppForDISet类,查看结果,说明userDao已经成功注入。
注入简单数据类型
需求:给BookDaoImpl注入一些简单数据类型的数据
参考引用数据类型的注入,我们可以推出具体的步骤为:
1.在BookDaoImpl类中声明对应的简单数据类型的属性
2.为这些属性提供对应的setter方法
3.在applicationContext.xml中配置
思考:
引用类型使用的是<property name="" ref=""/>,简单数据类型还是使用ref么?
ref是指向Spring的IOC容器中的另一个bean对象的,对于简单数据类型,没有对应的bean对象,
该如何配置?
步骤1:声明属性并提供setter方法
在BookDaoImpl类中声明对应的简单数据类型的属性,并提供对应的setter方法
1 public class BookDaoImpl implements BookDao {
2
3 private String databaseName;
4 private int connectionNum;
5
6 public void setConnectionNum(int connectionNum) {
7 this.connectionNum = connectionNum;
8 }
9
10 public void setDatabaseName(String databaseName) {
11 this.databaseName = databaseName;
12 }
13
14 public void save() {
15 System.out.println("book dao save
..."+databaseName+","+connectionNum);
16 }
17 }
步骤2:配置文件中进行注入配置
在applicationContext.xml配置文件中使用property标签注入
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl">
7 <property name="databaseName" value="mysql"/>
8 <property name="connectionNum" value="10"/>
9 </bean>
10 <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
11 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
12 <property name="bookDao" ref="bookDao"/>
13 <property name="userDao" ref="userDao"/>
14 </bean>
15 </beans>
名称 值
只要你用了value,你就可以不用管它的类型了,spring自动帮我们配置好了
说明:
value:后面跟的是简单数据类型,对于参数类型,Spring在注入的时候会自动转换,但是不能写成
1 <property name="connectionNum" value="abc"/>
这样的话,spring在将abc转换成int类型的时候就会报错。
步骤3:运行程序
运行AppForDISet类,查看结果,说明userDao已经成功注入。
注意:两个property注入标签的顺序可以任意。
对于setter注入方式的基本使用就已经介绍完了,
- 对于引用数据类型使用的是<property name="" ref=""/>
- 对于简单数据类型使用的是<property name="" value=""/>
构造器注入
环境准备
构造器注入也就是构造方法注入,学习之前,还是先准备下环境:
- 创建一个Maven项目
- pom.xml添加依赖
- resources下添加spring的配置文件
这些步骤和前面的都一致,大家可以快速的拷贝即可,最终项目的结构如下:
(1)项目中添加BookDao、BookDaoImpl、UserDao、UserDaoImpl、BookService和
BookServiceImpl类
1 public interface BookDao {
2 public void save();
3 }
4
5 public class BookDaoImpl implements BookDao {
6
7 private String databaseName;
8 private int connectionNum;
9
10 public void save() {
11 System.out.println("book dao save ...");
12 }
13 }
14 public interface UserDao {
15 public void save();
16 }
17 public class UserDaoImpl implements UserDao {
18 public void save() {
19 System.out.println("user dao save ...");
20 }
21 }
22
23 public interface BookService {
24 public void save();
25 }
26
27 public class BookServiceImpl implements BookService{
28 private BookDao bookDao;
29
30 public void setBookDao(BookDao bookDao) {
31 this.bookDao = bookDao;
32 }
33
34 public void save() {
35 System.out.println("book service save ...");
36 bookDao.save();
37 }
38 }
右击Constructor构造器生成
(2)resources下提供spring的配置文件
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
8 <property name="bookDao" ref="bookDao"/>
9 </bean>
10 </beans>
(3)编写AppForDIConstructor运行类,加载Spring的IOC容器,并从中获取对应的bean对象
1 public class AppForDIConstructor {
2 public static void main( String[] args ) {
3 ApplicationContext ctx = new
ClassPathXmlApplicationContext("applicationContext.xml");
4 BookService bookService = (BookService) ctx.getBean("bookService");
5 bookService.save();
6 }
7 }
构造器注入引用数据类型
接下来,在上面这个环境中来完成构造器注入的学习:
需求:将BookServiceImpl类中的bookDao修改成使用构造器的方式注入。
1.将bookDao的setter方法删除掉
2.添加带有bookDao参数的构造方法
3.在applicationContext.xml中配置
步骤1:删除setter方法并提供构造方法
在BookServiceImpl类中将bookDao的setter方法删除掉,并添加带有bookDao参数的构造方法
1 public class BookServiceImpl implements BookService{
2 private BookDao bookDao;
3
4 public BookServiceImpl(BookDao bookDao) {
5 this.bookDao = bookDao;
6 }
7
8 public void save() {
9 System.out.println("book service save ...");
10 bookDao.save();
11 }
12 }
步骤2:配置文件中进行配置构造方式注入
在applicationContext.xml中配置
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
8 <constructor-arg name="bookDao" ref="bookDao"/>
9 </bean>
10 </beans>
说明:
标签中
- name属性对应的值为构造函数中方法形参的参数名,必须要保持一致。
- ref属性指向的是spring的IOC容器中其他bean对象。
步骤3:运行程序
运行AppForDIConstructor类,查看结果,说明bookDao已经成功注入。
构造器注入多个引用数据类型
需求:在BookServiceImpl使用构造函数注入多个引用数据类型,比如userDao
1.声明userDao属性
2.生成一个带有bookDao和userDao参数的构造函数
3.在applicationContext.xml中配置注入
步骤1:提供多个属性的构造函数
在BookServiceImpl声明userDao并提供多个参数的构造函数
1 public class BookServiceImpl implements BookService{
2 private BookDao bookDao;
3 private UserDao userDao;
4
5 public BookServiceImpl(BookDao bookDao,UserDao userDao) {
6 this.bookDao = bookDao;
7 this.userDao = userDao;
8 }
9
10 public void save() {
11 System.out.println("book service save ...");
12 bookDao.save();
13 userDao.save();
14 }
15 }
步骤2:配置文件中配置多参数注入
在applicationContext.xml中配置注入
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
8 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
9 <constructor-arg name="bookDao" ref="bookDao"/>
10 <constructor-arg name="userDao" ref="userDao"/>
11 </bean>
12 </beans>
说明:这两个<contructor-arg>的配置顺序可以任意
步骤3:运行程序
运行AppForDIConstructor类,查看结果,说明userDao已经成功注入。
构造器注入多个简单数据类型
需求:在BookDaoImpl中,使用构造函数注入databaseName和connectionNum两个参数。
参考引用数据类型的注入,我们可以推出具体的步骤为:
1.提供一个包含这两个参数的构造方法
2.在applicationContext.xml中进行注入配置
步骤1:添加多个简单属性并提供构造方法
修改BookDaoImpl类,添加构造方法
1 public class BookDaoImpl implements BookDao {
2 private String databaseName;
3 private int connectionNum;
4
5 public BookDaoImpl(String databaseName, int connectionNum) {
6 this.databaseName = databaseName;
7 this.connectionNum = connectionNum;
8 }
9
10 public void save() {
11 System.out.println("book dao save
..."+databaseName+","+connectionNum);
12 }
13 }
步骤2:配置完成多个属性构造器注入
在applicationContext.xml中进行注入配置
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4 xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
5
6 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl">
7 <constructor-arg name="databaseName" value="mysql"/>
8 <constructor-arg name="connectionNum" value="666"/>
9 </bean>
10 <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
11 <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
12 <constructor-arg name="bookDao" ref="bookDao"/>
13 <constructor-arg name="userDao" ref="userDao"/>
14 </bean>
15 </beans>
说明:这两个<contructor-arg>的配置顺序可以任意
步骤3:运行程序
运行AppForDIConstructor类,查看结果
上面已经完成了构造函数注入的基本使用,但是会存在一些问题:
- 当构造函数中方法的参数名发生变化后,配置文件中的name属性也需要跟着变
- 这两块存在紧耦合,具体该如何解决?
在解决这个问题之前,需要提前说明的是,这个参数名发生变化的情况并不多,所以上面的还是比较
主流的配置方式,下面介绍的,大家都以了解为主。
方式一:删除name属性,添加type属性,按照类型注入
1 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl">
2 <constructor-arg type="int" value="10"/>
3 <constructor-arg type="java.lang.String" value="mysql"/>
4 </bean>
- 这种方式可以解决构造函数形参名发生变化带来的耦合问题
- 但是如果构造方法参数中有类型相同的参数,这种方式就不太好实现了
方式二:删除type属性,添加index属性,按照索引下标注入,下标从0开始
1 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl">
2 <constructor-arg index="1" value="100"/>
3 <constructor-arg index="0" value="mysql"/>
4 </bean>
- 这种方式可以解决参数类型重复问题
- 但是如果构造方法参数顺序发生变化后,这种方式又带来了耦合问题
介绍完两种参数的注入方式,具体我们该如何选择呢?
1. 强制依赖使用构造器进行,使用setter注入有概率不进行注入导致null对象出现
强制依赖指对象在创建的过程中必须要注入指定的参数
2. 可选依赖使用setter注入进行,灵活性强
可选依赖指对象在创建过程中注入的参数可有可无
3. Spring框架倡导使用构造器,第三方框架内部大多数采用构造器注入的形式进行数据初始化,相
对严谨
4. 如果有必要可以两者同时使用,使用构造器注入完成强制依赖的注入,使用setter注入完成可选
依赖的注入
5. 实际开发过程中还要根据实际情况分析,如果受控对象没有提供setter方法就必须使用构造器注
入
6. 自己开发的模块推荐使用setter注入
这节中主要讲解的是Spring的依赖注入的实现方式:
setter注入
简单数据类型
1 <bean ...>
2 <property name="" value=""/>
3 </bean>
引用数据类型
1 <bean ...>
2 <property name="" ref=""/>
3 </bean>
构造器注入
简单数据类型
1 <bean ...>
2 <constructor-arg name="" index="" type="" value=""/>
3 </bean>
引用数据类型
1 <bean ...>
2 <constructor-arg name="" index="" type="" ref=""/>
3 </bean>
依赖注入的方式选择上
- 建议使用setter注入
- 第三方技术根据情况选择