仓库地址

w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started quickly through a series of examples (github.com)

Chapter02-Spring依赖注入的方式

我们在Chapter00—2.2节依赖注入已经介绍了Spring的对象依赖注入的方式,在那个例子中,我们使用了字段的setter方法对字段进行了注入。在本章中,我们将介绍对象依赖注入的另外的方式,并提到一些关于依赖注入的注意点。

大致来说,依赖注入分为三种:

  1. 属性setter方法注入
  2. 字段注入
  3. 构造函数注入

为了 接下来的示例做准备,我们按照如下的代码结构顺序编写:

  1. 编写类Pen,表示一个笔类Pen
  2. 编写类Box,表示一个用于装Pen的盒子类Box
  3. 编写相关配置注入的代码
  4. 使用Spring验证代码注入

OK,首先编写类Pen做准备:

@Component
public class Pen {
public Pen() {
System.out.println("Pen 无参构造函数");
}
}

对于该类,我们使用@Component将其标记为Bean(PS:这里为了本章的主题,我们就不使用上一章的配置类和XML来声明了)。

一 属性setter方法注入

编写类BoxA类:

@Component
public class BoxA { private Pen pen; // 我们在属性setter方法上,使用@Autowire注解
// 目的是希望Spring容器框架能够让帮助我们注入该属性
@Autowired
public void setPen(Pen pen) {
this.pen = pen;
System.out.println("setter函数注入:" + pen);
} public void printPen() {
System.out.println(pen == null ? "BoxA没有Pen" : "BoxA有Pen:" + pen);
} }

对于该BoxA类,我们同样使用@Component标记为了Bean。此外,我们为其添加了Pen类型的字段pen,并编写了setter方法。在该方法上,我们添加了@Autowired注解,表明我们希望类型为Pen的属性pen能够由Spring为我们注入进来。

那么现在有一个问题,这个Pen的实例,是怎么来的呢?不难思考出来,就是我们一开头准备的@Component Pen

接下来编写验证代码:

@SpringBootApplication
public class Chapter02App {
public static void main(String[] args) {
ConfigurableApplicationContext context =
SpringApplication.run(Chapter02App.class, args);
// 我们只获取了BoxA的实例
Object boxA = context.getBean("boxA");
if (boxA instanceof BoxA) {
System.out.println("BoxA实例:" + box);
((BoxA) boxA).printPen(); // 却能够正确打印Pen实例
}
}
}

二 构造函数注入

编写BoxB类:

@Component
public class BoxB { private Pen pen; // 在构造函数上添加@Autowired注解,并且构造函数入参有Pen
@Autowired
public BoxB(Pen pen) {
this.pen = pen;
System.out.println("BoxB 有参构造函数,注入Pen实例:" + pen);
} public void printPen() {
System.out.println(pen == null ? "BoxB没有Pen" : "BoxB有Pen:" + pen);
}
}

对于该类,我们同样拥有Pen类型字段pen,但和setter方法注入不同的是,我们没有编写setter方法,而是显式编写了BoxB的构造函数,并且构造函数的入参就是Pen类的实例,然后在该构造函数上也同样添加了注解@Autowired。其意图其实和setter方法注入相同,都是希望Spring框架能够帮我们进行依赖注入。

对于验证的代码,我们只需要稍微修改以下:

@SpringBootApplication
public class Chapter02App {
public static void main(String[] args) {
ConfigurableApplicationContext context =
SpringApplication.run(Chapter02App.class, args);
// 我们只获取了BoxB的实例
Object boxB = context.getBean("boxB");
if (boxB instanceof BoxB) {
System.out.println("BoxB实例:" + boxB);
((BoxB) boxB).printPen(); // BoxB 能够正确打印Pen实例
}
}
}

三 字段注入

对于字段注入来说,就更简单了,BoxC:

@Component
public class BoxC { @Autowired
private Pen pen; public void printPen() {
System.out.println(pen == null ? "BoxC没有Pen" : "BoxC有Pen:" + pen);
}
}

没有构造函数,没有setter方法,只有无尽的怒火字段,然后字段上添加@Autowired注解。验证代码同上,不再赘述。

字段注入比起另外的两种两种方式简单的多,可能绝大多数的项目都会用这个字段,但本人将字段注入放在了第三个来讲,还是希望说一下字段注入的问题点。使用IDEA同学可能已经看到了,IDEA会提示:

Field injection is not recommended
不推荐使用字段注入

字段注入这么方便,为什么说不推荐呢?主要有以下几点:

  1. 基于字段的依赖注入在声明为final的字段上不起作用。
  2. 会与SpringIOC容器框架紧密耦合。因为private字段的原因,想要编写单元测试,就必须依赖Spring测试框架,否则你无法手动注入(除了使用反射,但是那样不久太麻烦了吗?)。

字段注入的问题还有其他的问题,可以自行搜索:Spring不推荐字段注入。

当然,如果一个项目自始自终都是在Spring框架中运行,也没有所谓的需要脱离Spring框架的地方,字段注入也并非不可。

扩展阅读:依赖注入的注意点

我们在上文已经提到了三种依赖注入的方式。那么读者有没有想过依赖注入需要注意什么呢?

编写类TestA、TestB:

// TestA类
@Component
public class TestA { // 字段注入
@Autowired
private TestB testB; public void printTestB() {
System.out.println("printTestB: " + this.testB);
}
}
// TestB类
@Component
public class TestB { @Autowired
private TestA testA; public void printTestA() {
System.out.println("printTestA: " + this.testA);
}
}

注意TestA类与TestB类相互以字段注入的方式注入到了另一个类中。接着我们编写测试代码:

@SpringBootApplication
public class Chapter01CycleTestApp {
public static void main(String[] args) {
ConfigurableApplicationContext context =
SpringApplication.run(Chapter01CycleTestApp.class, args);
TestA testA = context.getBean(TestA.class);
TestB testB = context.getBean(TestB.class);
System.out.println("从容器中获取的TestA实例:" + testA);
testA.printTestB(); System.out.println("从容器中获取的TestB实例:" + testB);
testB.printTestA();
}
}

那么,测试代码能够正常运行呢?思考一下,我们似乎陷入了循环依赖的场景了:

Spring容器创建TestA实例 -> 发现需要注入TestB实例 -> 创建TestB实例 -> 发现需要注入实例TestA -> 创建TestA实例 -> ...

似乎陷入了一直循环的情景。可是实际运行的时候,却正确输出了:

...
从容器中获取的TestA实例:TestA@4248ed58
printTestB: TestB@712ca57b
从容器中获取的TestB实例:TestB@712ca57b
printTestA: TestA@4248ed58 PS: 为了简洁的显示,我删去了包名

为什么会这样呢?实际上,Spring在初始化Bean的时候,并不是傻乎乎的按照上述的逻辑进行的,而是按照如下的大致流程:

  1. 准备创建TestA实例
  2. 发现TestA依赖TestB
  3. 查找TestB实例未果
  4. 先继续创建TestA实例,但是内部会标记该类实际还未注入依赖
  5. 准备创建TestB实例
  6. 发现TestB实例依赖TestA实例
  7. 查找TestA实例成功(只不过目前还没注入依赖而已),注入
  8. 扫描所有还未完全完成初始化的Bean,发现TestA还未注入依赖TestB
  9. 再次检查发现TestB已经有了,将其注入到TestA中

当然,这里只是简单的梳理一个流程,在Spring内部是很复杂的。

那么现在再来思考另外一个相互依赖注入的情况:构造函数注入。

// TestA 构造函数注入
@Component
public class TestA { // 字段注入
// @Autowired
private TestB testB; @Autowired
public TestA(TestB testB) {
this.testB = testB;
} public void printTestB() {
System.out.println("printTestB: " + this.testB);
}
}
// TestB构造函数注入
@Component
public class TestB { // @Autowired
private TestA testA; // 构造函数注入
@Autowired
public TestB(TestA testA) {
this.testA = testA;
} public void printTestA() {
System.out.println("printTestA: " + this.testA);
}
}

现在我们通过构造函数注入,执行测试代码看看会发生什么:

The dependencies of some of the beans in the application context form a cycle:

┌─────┐
| testA defined in file [xxx\chapter02\cycle\TestA.class]
↑ ↓
| testB defined in file [xxx\chapter02\cycle\TestB.class]
└─────┘

”当前应用程序中某些beans出现了循环依赖“。至于原因,请搜索关键词:Spring构造函数注入与setter注入

本章小结

在本章中,我们了解了Spring依赖注入的三种方式,并提到了循环依赖在不同注入方式下的区别。

至此,关于学习SpringBoot前的需要有的知识我们算是完成了一个简单的介绍的,接下来的文章将会基于SpringBoot框架,构建我们的Web服务。

极简SpringBoot指南-Chapter02-Spring依赖注入的方式的更多相关文章

  1. Spring依赖注入的方式、类型、Bean的作用域、自动注入、在Spring配置文件中引入属性文件

    1.Spring依赖注入的方式 通过set方法完成依赖注入 通过构造方法完成依赖注入 2.依赖注入的类型 基本数据类型和字符串 使用value属性 如果是指向另一个对象的引入 使用ref属性 User ...

  2. Spring 依赖注入的方式

    Spring 支持3中依赖注入的方式 1.属性注入  通过setter 方法注入Bean的属性或依赖的对象. <bean id = " " class = " &q ...

  3. 极简SpringBoot指南-Chapter01-如何用Spring框架声明Bean

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  4. 极简SpringBoot指南-Chapter00-学习SpringBoot前的基本知识

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  5. 极简SpringBoot指南-Chapter03-基于SpringBoot的Web服务

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  6. spring依赖注入的方式(一)

    为了方便类的管理,spring提供了依赖注入的思想:类的实例化不由程序员控制,而是交给sprig容器进行管理. spring提供了多种类型的注入方式---注解.xml注入: 1  注解注入 有两种:@ ...

  7. 极简SpringBoot指南-Chapter05-SpringBoot中的AOP面向切面编程简介

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  8. 极简SpringBoot指南-Chapter04-基于SpringBoot的书籍管理Web服务

    仓库地址 w4ngzhen/springboot-simple-guide: This is a project that guides SpringBoot users to get started ...

  9. Spring依赖注入:@Autowired,@Resource和@Inject区别与实现原理

    一.spring依赖注入使用方式 @Autowired是spring框架提供的实现依赖注入的注解,主要支持在set方法,field,构造函数中完成bean注入,注入方式为通过类型查找bean,即byT ...

随机推荐

  1. docker 安装Hive

    转自:https://www.cnblogs.com/upupfeng/p/13452385.html#%E9%83%A8%E7%BD%B2hive 使用docker快速搭建hive环境   记录一下 ...

  2. java 方法参数的执行顺序

    java方法的参数的执行顺序是从左到右还是从右到左呢? 写出一下测试程序: 1 import java.util.*; 2 import java.io.*; 3 public class Test ...

  3. 【MATLAB】常用命令快速入门,国赛加油

    矩阵运算 矩阵的基本生成 m1 = 1:5 % 生成行矩阵[1,2,3,4,5] m2 = 1:2:10 % 起点:步长:终点 [1,3,5,7,9] linspace(x1,x2,n) % 生成 n ...

  4. opencv入门系列教学(七)改变颜色空间、提取彩色对象

    ​ 0.序言 之前的博客里我们介绍了opencv在图像上的基本操作,下面我们来进行稍微深入一点的介绍,从这里开始我们可以发现opencv库能给我们带来的更多更有趣的功能.从现在开始,我们将逐步深入了解 ...

  5. Spring Boot +Vue 项目实战笔记(一):使用 CLI 搭建 Vue.js 项目

    前言 从这篇文章开始,就进入真正的实践了. 在前端项目开发中,我们可以根据实际情况不同程度地使用 Vue.利用 Vue CLI(或写成 vue-cli,即 Vue 脚手架)搭建出来的项目,是最能体现 ...

  6. VMware ESXi 7.0 U2 SLIC & Unlocker Intel NUC 专用镜像 202109 更新

    2021.08.31 更新:集成 "vmkusb-nic-fling" 和 "nvme-community",现在只有一个镜像. 2021.06.16 更新:集 ...

  7. k8s笔记0528-基于KUBERNETES构建企业容器云手动部署集群记录-2

    三.ETCD集群部署 类似于走zookeeper集群分布式协调服务,可做以key v形式存储在ETCD中. 官方链接:https://github.com/coreos/etcd 分布式kv存储,为分 ...

  8. Windows内核-7-IRP和派遣函数

    Windows内核-7-IRP和派遣函数 IRP以及派遣函数是Windows中非常重要的概念.IRP 是I/O Request Pocket的简称,意思是I/O操作的请求包,Windows中所有Use ...

  9. Python - 面向对象编程 - 使用 super() 的一些注意事项

    super() 详解 https://www.cnblogs.com/poloyy/p/15223443.html 多继承中使用 super() class A: def test(self): pr ...

  10. Linux 配置Maven(避免踩坑篇)

    前言:请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i 一.访问Maven官网下载压缩文件. 二.下载好的maven安装包放在磁盘的 /usr/local/ 目录下,如下图: 三.解压该压缩文 ...