0%

理解Spring中的IoC和DI

什么是IoC和DI

IoC(Inversion of Control 控制反转):是一种面向对象编程中的一种设计原则,用来降低计算机代码之间的耦合度。其基本思想是:借助于“第三方”实现具有依赖关系的对象之间的解耦。

DI(Dependence Injection 依赖注入):将实例变量传入到一个对象中去(Dependency injection means giving an object its instance variables)。

  • 控制反转是一种思想
  • 依赖注入是一种设计模式
  • IoC框架使用依赖注入作为实现控制反转的方式

为什么需要

在没有IoC之前,我们要在A类中使用B类,就要在A类中newB类的实例,这样A类和B类之间就出现了耦合。

1
2
3
public class A {
private B b = new B();
}

使用了IoC之后,我们就把实例化这样操作交给框架去帮我们做了。

Spring 中的IoC

容器是Spring的核心,Spring容器负责创建应用程序中的bean并通过DI来协调这些对象之间的关系。

Spring容器并不是只有一个,Spring自带多个容器的实现,可以归纳为两种不同的类型:

  1. bean工厂(BeanFactory),最简单的容器,提供基本的DI支持。
  2. 应用上下文(ApplicationContext),继承了BeanFactory,并提供应用框架级别的服务。

作为开发人员,我们需要告诉Spring哪些对象要作为bean装配到容器中,bean和bean之间的依赖关系。Spring提供了三种主要的装配机制:

  • 隐式的bean发现机制和自动装配
  • 在Java中进行显示配置
  • 在XML中进行显示配置

下面我们逐一介绍这三种机制。

自动装配bean

组件扫描:Spring会自动发现应用上下文中所创建的bean

@Component 注解表明该类会作为组件类,并告知Spring要为这个类创建bean。

1
2
3
4
@Component
public class Dog {

}

@ComponentScan 注解启用了组件扫描。

1
2
3
4
5
@Configuration
@ComponentScan
public class DemoApplication {

}

自动装配:Spring自动满足bean之间的依赖

@Autowired 注解可以作用在构造器、方法、属性上。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
@Component
public class Dog {
// 属性
@Autowired
private Cat cat;

// 构造器
// 从Spring 4.3开始,具有单个构造函数的类可以省略@Autowired注释
@Autowired
public Dog(Cat cat) {
this.cat = cat;
}

// 方法
@Autowired
public void setCat(Cat cat) {
this.cat = cat;
}
}

在Java中装配bean

组价配置:声明一个配置类,并在配置类中配置bean

@Configuration 注解表明这个类是配置类,我们可以在配置类下创建bean。

@bean 注解会告诉Spring这个方法将会返回一个对象,该对象要注册为Spring上下文中的bean。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
/**
* 普通类
*/
public class BaseBean {

public void p() {
System.out.println("Hello bean");
}
}



/**
* 配置类
*/
@Configuration
public class BeanConfig {

// 这个方法返回一个对象,Spring会把这个对象注册为bean
@Bean
public BaseBean getBaseBean() {
return new BaseBean();
}

}

组件注入:在配置类中把被依赖的组件注入另一个组件中

两种方式注入bean:

  1. 我们可以直接调用get方法,获取到对应的组件
  2. get方法中把被依赖的组件作为参数传入,Spring在调用这个方法时,会自动为你注入。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
/**
* 普通类
*/
public class BaseBean {

public void p() {
System.out.println("Hello bean");
}
}

/**
* 普通类
*/
public class UserBean {
private BaseBean baseBean;

public UserBean(BaseBean baseBean) {
this.baseBean = baseBean;
}
}



/**
* 配置类
*/
@Configuration
public class BeanConfig {

// 这个方法返回一个对象,Spring会把这个对象注册为bean
@Bean
public BaseBean getBaseBean() {
return new BaseBean();
}

/**
* 以下为两种注入bean的方法
*/
// 方法一:直接调用get方法
@Bean
public UseBean getUseBean() {
return new UseBean(getBaseBean());
}

// 方法二:当做参数传入,Spring将自动为你注入
@Bean
public UseBean getUseBean(BaseBean baseBean) {
return new UseBean(baseBean);
}

}

通常情况下我们都会使用方法二。

通过XML装配bean

尽管现在我们已经不怎么再使用XML装配bean,但在Spring刚刚出现的时候,XML是描述配置的主要方式,我们还是有必要了解一下的。

在使用JavaConfig的时候,我们创建了一个配置类来装配bean,而在XML配置中,我们需要创建一个XML文件,并且要以<beans>元素为根。

最为简单的Spring XML配置如下所示:

1
2
3
4
5
6
7
8
9
<?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 -->

</beans>

组件配置

以上文的BaseBean为例,我们在XML文件中把它声明为bean。

1
<bean id="baseBean" class="com.example.demo.BaseBean" />

组件注入

1
2
<bean id="useBean" class="com.example.demo.UseBean"
c:_="baseBean" />

XML的语法我就不再这里详述了,有兴趣的同学可以自行学习。

总结

本文我们简单介绍了Spring中的IoC,介绍了Spring中装配bean的三种方式:自动化配置,基于Java的显式配置以及基于XML的显式配置。这些技术都是为了描述Spring应用中的组件以及组件之间的关系。

一般来说我们都会使用自动化配置,尽量避免显式配置带来的维护成本。如果不得不使用显式配置的话,我们优先选择基于Java的配置,它比基于XML的配置更加强大、类型安全并且易于重构。

参考资料

控制反转(IoC)与依赖注入(DI)

Spring 实战

Spring 揭秘

本文首发于我的个人博客 https://chaohang.top

作者张小超

转载请注明出处

欢迎关注我的微信公众号 【超超不会飞】,获取第一时间的更新。