1.4 依赖注入

1.4.1 依赖注入的概念

依赖注入(Dependency Injection,简称DI)与控制反转(IoC)的含义相同,只不过这两个称呼是从两个角度描述的同一个概念。对于一个Spring初学者来说,这两种称呼很难理解,下面我们将通过简单的语言来描述这两个概念。

当某个Java对象(调用者)需要调用另一个Java对象(被调用者,即被依赖对象)时,在传统模式下,调用者通常会采用“new被调用者”的代码方式来创建对象,如图1-8所示。这种方式会导致调用者与被调用者之间的耦合性增加,不利于后期项目的升级和维护。

图1-8 调用者创建被调用者对象

在使用Spring框架之后,对象的实例不再由调用者来创建,而是由Spring容器来创建,Spring容器会负责控制程序之间的关系,而不是由调用者的程序代码直接控制。这样,控制权由应用代码转移到了Spring容器,控制权发生了反转,这就是Spring的控制反转。

从Spring容器的角度来看,Spring容器负责将被依赖对象赋值给调用者的成员变量,这相当于为调用者注入了它依赖的实例,这就是Spring的依赖注入,如图1-9所示。

图1-9 将被调用者对象注入调用者对象

1.4.2 依赖注入的实现方式

依赖注入的作用就是在使用Spring框架创建对象时,动态地将其所依赖的对象注入Bean组件中,其实现方式通常有两种,一种是属性setter方法注入,另一种是构造方法注入,具体介绍如下。

· 属性setter方法注入:指IoC容器使用setter方法注入被依赖的实例。通过调用无参构造器或无参静态工厂方法实例化Bean后,调用该Bean的setter方法,即可实现基于setter方法的依赖注入。

· 构造方法注入:指IoC容器使用构造方法注入被依赖的实例。基于构造方法的依赖注入通过调用带参数的构造方法来实现,每个参数代表着一个依赖。

了解了两种注入方式后,下面以属性setter方法注入的方式为例,讲解一下Spring容器在应用中是如何实现依赖注入的。

(1)在com.itheima.ioc包中,创建接口UserService,在接口中编写一个say()方法,如文件1-5所示。

文件1-5 UserService.java

1  package com.itheima.ioc;
2  public interface UserService {3
public void say();
4  }

(2)在com.itheima.ioc包中,创建UserService接口的实现类UserServiceImpl,在类中声明userDao属性,并添加属性的setter方法,如文件1-6所示。

文件1-6 UserServiceImpl.java

1  package com.itheima.ioc;
2  public class UserServiceImpl implements UserService {
3      // 声明UserDao属性
4      private UserDao userDao;
5      // 添加UserDao属性的setter方法,用于实现依赖注入
6      public void setUserDao(UserDao userDao) {
7          this.userDao = userDao;
8      }
9      // 实现的接口中方法
10     public void say() {
11         //调用userDao中的say()方法,并执行输出语句
12         this.userDao.say();
13         System.out.println("userService say hello World ! ");
14     }
15 }

(3)在配置文件applicationContext.xml中,创建一个id为userService的Bean,该Bean用于实例化UserServiceImpl类的信息,并将userDao的实例注入到userService中,其代码如下所示。

<! --添加一个id为userService的实例 -->
<bean id="userService" class="com.itheima.ioc.UserServiceImpl">
  <! -- 将id为userDao的Bean实例注入到userService实例中 -->
  <property name="userDao" ref="userDao" />
</bean>

在上述代码中,<property>是<bean>元素的子元素,它用于调用Bean实例中的setUserDao()方法完成属性赋值,从而实现依赖注入。其name属性表示Bean实例中的相应属性名,ref属性用于指定其属性值。

(4)在com.itheima.ioc包中,创建测试类TestDI,来对程序进行测试,编辑后如文件1-7所示。

文件1-7 TestDI.java

1  package com.itheima.ioc;
2  import org.springframework.context.ApplicationContext;
3  import
4     org.springframework.context.support.ClassPathXmlApplicationContext;
5  public class TestDI {
6      public static void main(String[] args) {
7          //1.初始化spring容器,加载配置文件
8          ApplicationContext applicationContext =
9             new ClassPathXmlApplicationContext("applicationContext.xml");
10         //2.通过容器获取UserService实例
11         UserService userService =
12           (UserService) applicationContext.getBean("userService");
13         //3.调用实例中的say()方法
14         userService.say();
15     }
16 }

(5)执行程序后,控制台的输出结果如图1-10所示。

图1-10 运行结果

从图1-10可以看出,使用Spring容器通过UserService实现类中的say()方法,调用了UserDao实现类中的say()方法,并输出了结果。这就是Spring容器属性setter注入的方式,也是实际开发中最为常用的一种方式。