Spring Bean装配笔记

Spring中的Bean是一个很重要的概念。Spring作为一个Bean容器,它可以管理对象和对象之间的依赖关系,我们不需要自己建立对象,把这部分工作全部转交给容器完成,具有低耦合,对代码没有侵略性。

目前本人的水平还不足以透彻的分析Spring是如何来构建Bean的装配的概念的,这里仅来记录总结Spring中多种Bean的装配方式。

Spring的配置方式有Java类配置形式与XML配置两种形式。大致提一下,不赘述,如下:

// Java类config形式,需要@Configuration注解来声明
@Configuration
public class MyConfig {
}
// 类路径下的XML形式
<?xml version="1.0" encoding="UTF-8"?>
<beans ...>
...
</beans>

Spring提供了三种装配机制:

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

隐式的bean发现机制和自动装配

隐式bean发现与自动装配即我们完全不必在配置类或者配置文件中定义bean属性(但是这两者必要提供一种)。 如何定义一个Java类是bean?我们可以在类上使用@Component注解声明一个bean。

首先定义一个接口:Playable

public interface Playable {
void play();
}

接下来,我们定一个CD类,实现该接口,并添加@Component注解

@Component
public class CD implements Playable {
public void play() {
System.out.println("CD is playing...");
}
}

如此定义,即可表明CD是一个bean。接下来如何开启扫描?

如果使用Java类配置的方式来配置Spring,我们可以使用@ComponentScan注解在配置类上,告诉Spring开启了组建扫描。并且进行形如如下的bean配置:

@Configuration
// 务必注意此处的自动扫描的基础包
// 因为通常配置类与其他类的包在不同的地方
// 默认的包路径是当前配置类所在包下
@ComponentScan(backPakages = "xxx")
public class MyConfig {
}

如果使用XMl来定义,则如下定义:

<beans>
<context:component-scan/>
</beans

接下来编写测试类:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = MyConfig.class)
public class MyConfigTest {
@Autowired
Playable playable; // 运行时,将扫描到的在容器中的CD bean装配到这里的playable
@Test
public void checkNULL() {
Assert.assertNotNull(playable);
}
}
// 结果自然是PASS

这里在配置类中我们完全没有定义任何的bean,Spring自动为我们扫描出对应的组件进行自动装配(Autowired)到测试类中。

CD这个类过于简单了,现在重构CD,使其拥有一个title属性,用以表示CD的名字。如下:

@Component
public class CD implements Playable {
private String title; public CD(String title) {
this.title = title;
} @Override
public void play() {
System.out.println("CD:" + title + " is playing...");
}
}

如果我们再次运行测试代码,会发现报错:

Unsatisfied dependency expressed through constructor parameter 0;// 大意是提示我们,CD这个类没有默认的参数个数为0的构造器。这里我们大致可以猜测,Spring在进行普通的构建对象时,是调用的该类的默认构造函数,在Java中我们知道,在一个类中若定义了任意形式的构造函数,原先的默认无参构造函数自然失效,而我们定义了有参数的构造函数,所以这里Spring调用不了无参构造函数,顾不能为我们构造这个bean。解决办法就是添加一个无参构造函数。

但是问题还没有彻底解决,这里Spring只是为我们创建了一个title没有初始化的CD实例对象,我们应该如何去初始化这个title呢?注入的是一个普通的对象,我们同样可以使用@Autowired注入,但是这里只是一个字面量String,如何注入?其实我们可以采用一种更加直观的方式来注入——显式配置

通过显式配置

配置类形式如下:

@Configuration
@ComponentScan(basePackages = "zhen")
public class MyConfig {
@Bean
Playable cd() {
return new CD("JayChou");
}
}

上面通过@Bean来声明下面我们将要定义一个bean,紧接着定义一个方法,返回值为Playable(这里没有严格的要求是接口类还是本身实现类,满足语法即可),方法名即为该bean对应的id,参数虽没有定义,但是不代表不能有,这里可以注入其它的bean。方法体中的内容自然是返回具体的实现类了,然而这里就很灵活,我们可以将字符串通过这里构造函数传入,假如我们的CD类中如果有setTitle方法,甚至还可以像下面这样:

    @Bean
Playable cd() {
CD cd = new CD("hello");
cd.setTitle("JayChou");
return cd;
}

配置文件XML如下:

<beans ...>
<bean id="cd" class="zhen.CD">
<constructor-arg value="JayChou" />
</bean>
</beans>

对比上面的两种显示注入的用法,对应也是很清楚的,Java类中的方法名即为bean id,返回对象对应的类即为XML中的class属性。同时,根据构造函数参数类型的不同,也有不同的形式,这里本篇笔记不多提,以后会有相关的笔记探讨的。

但是请注意,如果Java类配置文件中已经有一个@Bean,同时还启动了自动扫描,在原先的组件类上添加了@Component注解,Spring是会我们创建两个同为CD类实例bean的。如下:

// 注意已经添加了@Component注解
@Component
public class CD implements Playable {
private String title = "Default Title";
// 定义一个可以设置的title的方法
public void setTitle(String title) {
this.title = title;
} @Override
public void play() {
System.out.println("CD:" + title + " is playing...");
}
} //配置类中开启了组建扫描,且也定义了一个bean
@Configuration
@ComponentScan(basePackages = "zhen")
public class MyConfig {
@Bean
Playable cd() {
CD cd = new CD();
cd.setTitle("JayChou");
return cd;
}
}

在测试类不变的情况下,运行测试代码,会报错:

No qualifying bean of type 'zhen.component.Playable' available: expected single matching bean but found 2: CD,cd

这行报错告诉我们,发现了两个bean:CD cd都满足Playable,都可以注入到此处。 (注意,如果一个类有@Component组件注解,该类的bean id默认为类名首字母小写,这里CD由于其本身两个字母都是大写,Spring所以给其的默认名没有将首字母小写,一定注意命名特殊性) 到底该选谁呢?如果我们不指定,Spring也不能为我们做主。如果我们将测试类中的Playable playable改为Playable cd 或者 CD,如下:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = MyConfig.class)
public class MyConfigTest {
@Autowired
// 为了配合下面输出title进行验证
// 这里我将类型改为了CD,且CD类中的私有title我修改为了公有
CD cd;
// CD CD;
@Test
public void checkNULL() {
System.out.println(cd.title); // 输出 “CD:JayChou is playing...”
// System.out.println(CD.title); // 输出 "CD:Default String is playing..." // 下面均通过
Assert.assertNotNull(cd);
// Assert.assertNotNull(CD);
}
}

由上面的现象,我们可以推测Spring在注入的时候,首先根据@Autowired下面的引用名来查找对应类型的bean id,如果没有,再找同类型的bean(这里要补充一下,@Autowired是根据类型来匹配注入的)

以上笔记大致总结了一下Spring bean装配问题,然而还有问题没有解决:

上面提到过的报错,Spring如果找到了多个同类型的bean会提示报错,在没有声明的情况下,Spring不知道到底改选哪一个,这就是自动装配的歧义性问题。

以上两个问题,我会继续做笔记的。

Spring Bean装配笔记的更多相关文章

  1. Spring Bean装配详解(五)

    装配 Bean 的概述 前面已经介绍了 Spring IoC 的理念和设计,这一篇文章将介绍的是如何将自己开发的 Bean 装配到 Spring IoC 容器中. 大部分场景下,我们都会使用 Appl ...

  2. Spring Bean装配(下)——注解

    @Repository,@Service,@Controller这三个注解是基于component定义的注解 component-scan:组件扫描 base-package:扫描这个下的所有类 &l ...

  3. Spring Bean装配(上)

    Bean:在spring的IOC里面,把配置到IOC容器里面的实体或者是对象都称为Bean Bean配置项 Bean的作用域 Bean的生命周期 Bean的自动装配 Resources&Res ...

  4. Spring入门篇——第4章 Spring Bean装配(下)

    第4章 Spring Bean装配(下) 介绍Bean的注解实现,Autowired注解说明,基于java的容器注解说明,以及Spring对JSR支持的说明 4-1 Spring Bean装配之Bea ...

  5. Spring入门篇——第3章 Spring Bean装配(上)

    第3章 Spring Bean装配(上) 介绍Bean的作用域.生命周期.Aware接口.自动装配和Resource等内容. 3-1 Spring Bean装配之Bean的配置项及作用域 从上至下依次 ...

  6. Spring Bean装配学习

    解释:所谓装配就是把一个类需要的组件给它设置进去,英文就是wire,wiring:注解Autowire也叫自动装配. 目前Spring提供了三种配置方案: 在XML中进行显式的配置 在Java中进行显 ...

  7. spring Bean装配的几种方式简单介绍

    Spring容器负责创建应用程序中的bean同时通过ID来协调这些对象之间的关系.作为开发人员,我们需要告诉Spring要创建哪些bean并且如何将其装配到一起. spring中bean装配有两种方式 ...

  8. Spring Bean装配方式

    Spring装配机制 在xml中进行显示配置 在Java中进行显示配置 隐式bean发现机制和自动装配 自动化装配bean 组件扫描(component scanning),Spring会自动发现应用 ...

  9. Spring课程 Spring入门篇 4-8 Spring bean装配之基于java的容器注解说明--基于泛型的自动装配

    1 解析 1.1 什么是泛型? 1.2 泛型有什么作用? 1.3 泛型装配样式? 2 代码演练 2.1 泛型应用 1 解析 1.1 什么是泛型? Java泛型设计原则:只要在编译时期没有出现警告,那么 ...

随机推荐

  1. 刷题-力扣-剑指 Offer II 055. 二叉搜索树迭代器

    剑指 Offer II 055. 二叉搜索树迭代器 题目链接 来源:力扣(LeetCode) 链接:https://leetcode-cn.com/problems/kTOapQ 著作权归领扣网络所有 ...

  2. tf.app.run() 运行结束时,报错:SystemExit exception: no description

    环境:Python3.6.6 + tensorflow-gpu 源码如下: import tensorflow as tf def main(): print("hello tf.app.r ...

  3. python--接口自动化经常用到的pytest框架

    pytest常用的方法和原理 1.pytest的原理 pytest插件基于pluggy模块:pluggy有三个重要概念:HookspecMarker(用来定义hook函数),HookimplMarke ...

  4. cmd(命令行)超好用的技巧,很不错的打开方式

    超快速打开管理cmd widows + x 按a 直接打开文件位置,在地址栏输入cmd 地址----直接cmd打开到所在文件位置 ex:cmd D:\work cd ../../../ 返回上几层的方 ...

  5. k8s 存活探针,滚动更新

    文章原文 存活探针 Kubelet使用liveness probe(存活探针)来确定何时重启容器.例如,当应用程序处于运行状态但无法做进一步操作,liveness探针将捕获到deadlock,重启处于 ...

  6. 20201219 u,v,w

    开考前刚起床,所以一边考一边吃饭,然后整场都很迷... A. u 考场 半天才搞懂"下三角区域"指哪个区域,手模样例确认后打了 \(O(qn^2)\) 的裸暴力,然后就不会做了. ...

  7. 洛谷P3768 简单的数学题解题报告

    $$\begin{eqnarray}&\sum_{i=1}^{n}\sum_{j=1}^{n}ij\gcd(i,j)\\&\sum_{d=1}^{n}\sum_{i=1}^{n}\su ...

  8. Git 系列教程(3)- 初次运行 Git 前的配置

    前言 直接搬官网教程,再修改下,先啰嗦可以直接看以前的文章 Window初始化Git环境 https://www.cnblogs.com/poloyy/p/12185132.html Linux初始化 ...

  9. 乐肴2.0环境改名后,需要删除原来的软链接(public/storage)

    rm -rf public/storage php artisan storage:link

  10. 如何在C#中打开和读取EXCEL文件

    这篇文章向您展示如何在C#Windows Forms Application中使用ExcelDataReader,ExcelDataReader.DataSet打开和读取Excel文件.创建一个新的W ...