• Java开发学习(六)----DI依赖注入之setter及构造器注入解析


    一、DI依赖注入

    首先来介绍下Spring中有哪些注入方式?

    我们先来思考

    • 向一个类中传递数据的方式有几种?

      • 普通方法(set方法)

      • 构造方法

    • 依赖注入描述了在容器中建立bean与bean之间的依赖关系的过程,如果bean运行需要的是数字或字符串呢?

      • 引用类型

      • 简单类型(基本数据类型与String)

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

    • setter注入

      • 简单类型

      • 引用类型

    • 构造器注入

      • 简单类型

      • 引用类型

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

    二、setter注入

    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"/>

    2.1 环境准备

    环境准备:

    • 创建一个Maven项目

    • pom.xml添加依赖

    • resources下添加spring的配置文件

    最终项目的结构如下:

    (1)项目中添加BookDao、BookDaoImpl、UserDao、UserDaoImpl、BookService和BookServiceImpl类

    public interface BookDao {
        public void save();
    }
    ​
    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println("book dao save ...");
        }
    }
    public interface UserDao {
        public void save();
    }
    public class UserDaoImpl implements UserDao {
        public void save() {
            System.out.println("user dao save ...");
        }
    }
    ​
    public interface BookService {
        public void save();
    }
    ​
    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
    ​
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    ​
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }

    (2)resources下提供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"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <property name="bookDao" ref="bookDao"/>
        </bean>
    </beans>

    (3)编写AppForDISet运行类,加载Spring的IOC容器,并从中获取对应的bean对象

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

    2.2 注入引用数据类型

    需求:在bookServiceImpl对象中注入userDao

    1.在BookServiceImpl中声明userDao属性

    2.为userDao属性提供setter方法

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

    步骤1:声明属性并提供setter方法

    在BookServiceImpl中声明userDao属性,并提供setter方法

    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
        private UserDao userDao;
        
        public void setUserDao(UserDao userDao) {
            this.userDao = userDao;
        }
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    ​
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
            userDao.save();
        }
    }
    步骤2:配置文件中进行注入配置

    在applicationContext.xml配置文件中使用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.itheima.dao.impl.BookDaoImpl"/>
        <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <property name="bookDao" ref="bookDao"/>
            <property name="userDao" ref="userDao"/>
        </bean>
    </beans>
    步骤3:运行程序

    运行AppForDISet类,查看结果,说明userDao已经成功注入。

    2.3 注入简单数据类型

    需求:给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;
    ​
        public void setConnectionNum(int connectionNum) {
            this.connectionNum = connectionNum;
        }
    ​
        public void setDatabaseName(String databaseName) {
            this.databaseName = databaseName;
        }
    ​
        public void save() {
            System.out.println("book dao save ..."+databaseName+","+connectionNum);
        }
    }
    步骤2:配置文件中进行注入配置

    在applicationContext.xml配置文件中使用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.itheima.dao.impl.BookDaoImpl">
            <property name="databaseName" value="mysql"/>
            <property name="connectionNum" value="10"/>
        </bean>
        <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <property name="bookDao" ref="bookDao"/>
            <property name="userDao" ref="userDao"/>
        </bean>
    </beans>

    说明:

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

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

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

    步骤3:运行程序

    运行AppForDISet类,查看结果,说明userDao已经成功注入。

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

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

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

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

    三、构造器注入

    3.1 环境准备

    构造器注入也就是构造方法注入,还是先准备下环境:

    • 创建一个Maven项目

    • pom.xml添加依赖

    • resources下添加spring的配置文件

    这些步骤和前面的都一致,大家可以快速的拷贝即可,最终项目的结构如下:

    (1)项目中添加BookDao、BookDaoImpl、UserDao、UserDaoImpl、BookService和BookServiceImpl类

    public interface BookDao {
        public void save();
    }
    ​
    public class BookDaoImpl implements BookDao {
        
        private String databaseName;
        private int connectionNum;
        
        public void save() {
            System.out.println("book dao save ...");
        }
    }
    public interface UserDao {
        public void save();
    }
    public class UserDaoImpl implements UserDao {
        public void save() {
            System.out.println("user dao save ...");
        }
    }
    ​
    public interface BookService {
        public void save();
    }
    ​
    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
    ​
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    ​
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }

    (2)resources下提供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"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <property name="bookDao" ref="bookDao"/>
        </bean>
    </beans>

    (3)编写AppForDIConstructor运行类,加载Spring的IOC容器,并从中获取对应的bean对象

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

    3.2 构造器注入引用数据类型

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

    需求:将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:配置文件中进行配置构造方式注入

    在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.itheima.dao.impl.BookDaoImpl"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <constructor-arg name="bookDao" ref="bookDao"/>
        </bean>
    </beans>

    说明:

    标签<constructor-arg>

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

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

    步骤3:运行程序

    运行AppForDIConstructor类,查看结果,说明bookDao已经成功注入。

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

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

    1.声明userDao属性

    2.生成一个带有bookDao和userDao参数的构造函数

    3.在applicationContext.xml中配置注入

    步骤1:提供多个属性的构造函数

    在BookServiceImpl声明userDao并提供多个参数的构造函数

    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
        private UserDao userDao;
    ​
        public BookServiceImpl(BookDao bookDao,UserDao userDao) {
            this.bookDao = bookDao;
            this.userDao = userDao;
        }
    ​
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
            userDao.save();
        }
    }
    步骤2:配置文件中配置多参数注入

    在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.itheima.dao.impl.BookDaoImpl"/>
        <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <constructor-arg name="bookDao" ref="bookDao"/>
            <constructor-arg name="userDao" ref="userDao"/>
        </bean>
    </beans>

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

    步骤3:运行程序

    运行AppForDIConstructor类,查看结果,说明userDao已经成功注入。

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

    需求:在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中进行注入配置

    <?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">
            <constructor-arg name="databaseName" value="mysql"/>
            <constructor-arg name="connectionNum" value="666"/>
        </bean>
        <bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
        <bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
            <constructor-arg name="bookDao" ref="bookDao"/>
            <constructor-arg name="userDao" ref="userDao"/>
        </bean>
    </beans>

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

    步骤3:运行程序

    运行AppForDIConstructor类,查看结果

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

    • 当构造函数中方法的参数名发生变化后,配置文件中的name属性也需要跟着变,因为是形参的名字。

    • 这两块存在紧耦合,具体该如何解决?

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

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

    <bean id="bookDao" class="com.itheima.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.itheima.dao.impl.BookDaoImpl">
        <constructor-arg index="1" value="100"/>
        <constructor-arg index="0" value="mysql"/>
    </bean>
    • 这种方式可以解决参数类型重复问题

    • 但是如果构造方法参数顺序发生变化后,这种方式又带来了耦合问题

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

    1. 强制依赖使用构造器进行,使用setter注入有概率不进行注入导致null对象出现

      • 强制依赖指对象在创建的过程中必须要注入指定的参数

    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="" index="" type="" value=""/>
        </bean>
      • 引用数据类型

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

      • 建议使用setter注入

      • 第三方技术根据情况选择

     



    如果您觉得阅读本文对您有帮助,请点一下“推荐”按钮,您的“推荐”将是我最大的写作动力!欢迎各位转载,但是未经作者本人同意,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    DSP28335模块配置模板系列——定时器中断配置模板
    Redis之淘汰策略
    《Scratch等级考试(1~4级)历届真题解析》专栏总目录
    Sql文件导入数据库-保姆级教程
    GAN生成漫画脸
    基于springboot+vue的戒毒所人员管理系统毕业设计源码251514
    L83.linux命令每日一练 -- 第11章 Linux系统管理命令 -- ipcs和ipcrm
    redis学习
    12. Java异常及异常处理处理
    银河麒麟服务器x86安装qemu虚拟机,并安装windows server 2019
  • 原文地址:https://www.cnblogs.com/xiaoyh/p/16311038.html