《Spring实战》读书笔记-第2章 装配Bean

《Spring实战》是学习Spring框架的一本非常经典的书籍,之前阅读了这本书,只是在书本上写写画画,最近整理了一下《Spring实战》的读书笔记,通过博客的方式进行记录分享。俗话说,好记性不如烂笔头,把学习到的知识记录下来,方便温故知新,让更多的读者可以学习到有关Spring框架的知识。

| 序号 | 内容 | 链接地址 |

| — | — | — |

| 1 | 《Spring实战》读书笔记-第1章 Spring之旅 | https://blog.csdn.net/ThinkWon/article/details/103097364 |

| 2 | 《Spring实战》读书笔记-第2章 装配Bean | https://blog.csdn.net/ThinkWon/article/details/103527675 |

| 3 | 《Spring实战》读书笔记-第3章 高级装配 | https://blog.csdn.net/ThinkWon/article/details/103536621 |

| 4 | 《Spring实战》读书笔记-第4章 面向切面的Spring | https://blog.csdn.net/ThinkWon/article/details/103541166 |

| 5 | 《Spring实战》读书笔记-第5章 构建Spring Web应用程序 | https://blog.csdn.net/ThinkWon/article/details/103550083 |

| 6 | 《Spring实战》读书笔记-第6章 渲染Web视图 | https://blog.csdn.net/ThinkWon/article/details/103559672 |

| 7 | 《Spring实战》读书笔记-第7章 Spring MVC的高级技术 | https://blog.csdn.net/ThinkWon/article/details/103562467 |

文章目录

    • 2.1 Spring配置的可选方案
  • 2.2 自动化装配bean

  • 2.3 通过Java代码装配bean

  • 2.4 通过XML装配bean

  • 2.5 导入和混合配置

  • 2.6 小结

本章内容:

  • 声明bean

  • 构造器注入和Setter方法注入

  • 装配bean

  • 控制bean的创建和销毁

在Spring中,对象无需自己查找或创建与其所关联的其他对象。相反,容器负责把需要相互协作的对象引用赋予各个对象。例如,一个订单管理组件需要信用卡认证组件,但它不需要自己创建信用卡认证组件。订单管理组件只需要表明自己两手空空,容器就会主动赋予它一个信用卡认证组件。

创建应用对象之间协作关系的行为通常称为装配(wiring),这也是依赖注入(DI)的本质。在本章我们将介绍使用Spring装配 bean的基础知识。因为DI是Spring的最基本要素,所以在开发基于Spring的应用时,你随时都在使用这些技术。

在Spring中装配bean有多种方式。作为本章的开始,我们先花一点时间来介绍一下配置Spring容器最常见的三种方法。

2.1 Spring配置的可选方案


如第1章中所述,Spring容器负责创建应用程序中的bean并通过DI来协调这些对象之间的关系。但是,作为开发人员,你需要告诉Spring要创建哪些bean并且如何将其装配在一起。当描述bean如何进行装配时,Spring具有非常大的灵活性,它提供了三种主要的装配机制:

  • 在XML中进行显式配置。

  • 在Java中进行显式配置。

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

乍看上去,提供三种可选的配置方案会使Spring变得复杂。每种配置技术所提供的功能会有一些重叠,所以在特定的场景中,确定哪种技术最为合适就会变得有些困难。但是,不必紧张——在很多场景下,选择哪种方案很大程度上就是个人喜好的问题,你尽可以选择自己最喜欢的方式。

Spring有多种可选方案来配置bean,这是非常棒的,但有时候你必须要在其中做出选择。

这方面,并没有唯一的正确答案。你所做出的选择必须要适合你和你的项目。而且,谁说我们只能选择其中的一种方案呢?Spring的配置风格是可以互相搭配的,所以你可以选择使用XML装配一些bean,使用Spring基于Java的配置(JavaConfig)来装配另一些bean,而将剩余的bean让Spring去自动发现。

即便如此,我的建议是尽可能地使用自动配置的机制。显式配置越少越好。当你必须要显式配置bean的时候(比如,有些源码不是由你来维护的,而当你需要为这些代码配置bean的时候),我推荐使用类型安全并且比XML更加强大的JavaConfig。最后,只有当你想要使用便利的XML命名空间,并且在JavaConfig中没有同样的实现时,才应该使用XML。

在本章中,我们会详细介绍这三种技术并且在整本书中都会用到它们。现在,我们会尝试一下每种方法,对它们是什么样子的有一个直观的印象。作为Spring配置的开始,我们先看一下Spring的自动化配置。

2.2 自动化装配bean


Spring从两个角度来实现自动化装配:

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

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

二者结合使用,能够将显示配置降低到最少。

首先来一个示例:CD与CD播放器。

如果你不将CD插入到CD播放器中,那么CD播放器其实是没有太大用处的。

定义CD接口:

package com.springinaction;

public interface CompactDisc {

void play();

}

CD的一个实现类,使用@Component告知Spring要为这个类创建bean。

package com.springinaction;

import org.springframework.stereotype.Component;

@Component

public class SgtPeppers implements CompactDisc {

private String title = “Sgt. Pepper’s Lonely Hearts Club Band”;

private String artist = “The Beatles”;

@Override

public void play() {

System.out.println("Playing " + title + " by " + artist);

}

}

使用基于Java的Spring配置,使用@ComponentScan默认会扫描与配置类相同的包。

package com.springinaction;

import org.springframework.context.annotation.*;

@Configuration

@ComponentScan

public class CDPlayerConfig {

}

如果要是用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”

xmlns:Context = “http://www.springframework.org/schema/context”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd

http://www.springframework.org/schema/context

http://www.springframework.org/schema/context/spring-context.xsd">

<Context:component-scan base-package=“com.springinaction” />

使用JUnit进行测试,测试CD的实现类是否被Spring自动创建

package com.springinaction;

import static org.junit.Assert.*;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.test.context.ContextConfiguration;

import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(classes=CDPlayerConfig.class)

public class CDPlayerTest {

@Autowired

private CompactDisc sgtPeppers;

@Test

public void cdShouldNotBeNull(){

assertNotNull(sgtPeppers);

}

}

测试成功

img

此例展示了Spring创建bean的过程以及测试,其中SgtPeppers的bean并没有明确设置ID,但Spring会自动指定,默认为sgtPeppers,首字母为小写。如果想为这个bean设置不同的ID,可以用以下方式:

@Component(“lonelyHeartsClub”)

public class SgtPeppers implements CompactDisc {

}

到现在,我们没有为@ComponentScan设置任何属性。这意味这,按照默认规则,它会以配置类所在的包作为基础包(basepackage)来扫描组件。

如果我们想要将配置类放在单独的包中,使其与其他的应用代码区分开来。那么可用以下方式设置:

@Configuration

@ComponentScan(“com.springinaction”)

public class CDPlayerConfig { }

或者更加清晰地表明所设置的是基础包

@Configuration

@ComponentScan(basePackages=“com.springinaction”)

public class CDPlayerConfig { }

也可以设置多个基础包

@Configuration

@ComponentScan(basePackages={“com.springinaction”, “video”})

public class CDPlayerConfig { }

除了以上的方式还可以将其指定为包中所包含的类或接口:

@Configuration

@ComponentScan(basePackageClasses={CDPlayer.class, DVDPlayer.class})

public class CDPlayerConfig { }

这些类所在的包将会作为组件扫描的基础包。

系统中大部分类还是存在相互依赖的现象,所以我们就需要了解自动装配。

简单来说,自动装配就是让Spring自动满足bean依赖的一种方法,在满足依赖的过程中,会在Spring应用上下文中寻找匹配某个bean需求的其他bean。为了声明要进行自动装配,我们可以借助Spring的@Autowired注解

比如实现CD播放器的类,播放器接口是有一个play方法

package com.springinaction;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.stereotype.Component;

@Component

public class CDPlayer implements MediaPlayer {

private CompactDisc compactDisc;

@Autowired

public CDPlayer(CompactDisc compactDisc){

this.compactDisc = compactDisc;

}

@Override

public void play() {

compactDisc.play();

}

}

以上使用的是自动装配构造器,还能用在属性Setter方法上

@Autowired

public void setCompactDisc(CompactDisc compactDisc){

this.compactDisc = compactDisc;

}

如果没有匹配的bean,那么在应用上下文创建时,Spring会抛出一个异常。为了避免异常的出现,你可以将@Autowired的required属性设置为false:

@Autowired(required=false)

public CDPlayer(CompactDisc compactDisc){

this.compactDisc = compactDisc;

}

如果有多个bean都能满足依赖关系的话,Spring将会抛出一个异常。

另:@Autowired可以用@Inject代替

现在,我们已经在CDPlayer的构造器中添加了@Autowired注解,Spring将把一个可分配给CompactDisc类型的bean自动注入进来,为了验证,我们进行测试

package com.springinaction;

import static org.junit.Assert.*;

import org.junit.Test;

import org.junit.Rule;

import org.junit.contrib.java.lang.system.SystemOutRule;

import org.junit.runner.RunWith;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.test.context.ContextConfiguration;

import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(classes=CDPlayerConfig.class)

public class CDPlayerTest {

@Rule

public final SystemOutRule log = new SystemOutRule().enableLog();

@Autowired

private CompactDisc sgtPeppers;

@Autowired

private CDPlayer cdPlayer;

@Test

public void cdShouldNotBeNull(){

assertNotNull(sgtPeppers);

}

@Test

public void play(){

cdPlayer.play();

assertEquals(

"Playing Sgt. Pepper’s Lonely Hearts Club Band " +

“by The Beatles\n”,

log.getLog()

);

}

}

测试成功img

2.3 通过Java代码装配bean


当自动化的方案行不通的时候,就必须采用显示装配的方式。显示装配有两种可选方案:JavaConfig和XML。其中JavaConfig是更好的方案,因为它更为强大、类型安全并且对重构友好。

继续上面的示例代码,首先创建配置类

package com.springinaction;

import org.springframework.context.annotation.*;

@Configuration

public class CDPlayerConfig {

}

@Configuration注解表明这个类是一个配置类,该类应该包含在Spring应用上下文中如何创建bean的细节。

我们移除之前的@ComponentScan,使用显示配置

@Bean

public CompactDisc sgtPeppers(){

return new SgtPeppers();

}

@Bean注解会告诉Spring这个方法将会返回一个对象,该对象要注册为Spring应用上下文中的bean。方法体中包含了最终产生bean实例的逻辑。

默认情况下,bean的ID与带有@Bean注解的方法名一样。在本例中,bean的名字将会是sgtPeppers。也可以重命名

@Bean(name=“lonelyHeartsClubBand”)

public CompactDisc sgtPeppers(){

return new SgtPeppers();

}

下面我们将CD注入到CDPlayer中

@Bean

public CDPlayer cdPlayer(CompactDisc compactDisc){

return new CDPlayer(compactDisc);

}

需要注意的有两点:

  1. 构造器中不能用new创建的对象(这个对象的类是已经在Spring中被声明的),因为Spring会拦截所有对已声明对象的调用,并确保直接返回该方法所创建的bean。这是由于Spring所创建的bean都是单例的。

  2. compactDisc会在Spring中寻找已经实现CompactDisc的bean。

2.4 通过XML装配bean


先来看最简单的SpringXML配置

<?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

http://www.springframework.org/schema/context">

借助Spring Tool Suite创建XML配置文件创建和管理Spring XML配置文件的一种简便方式是使用Spring Tool Suite(https://spring.io/tools/sts)。

我们来简单声明一个bean

下面我们来看一些特征

  1. 不再需要直接负责创建SgtPeppers的实例,而在基于JavaConfig的配置中需要。当Spring发现这个元素时,它将会调用SgtPeppers的默认构造器来创建bean。

  2. 这个bean中,我们将bean的类型以字符串的形式设置了class属性中。没有类型检查。

XML声明DI时,会有很多种可选的配置方案和风格。具体到构造器注入,有两种基本的配置方案可供选择:

  • <constructor-arg>元素

  • 使用Spring3.0所引入的c-命名空间

先来看一下他们各组如何注入bean引用

构造器注入bean引用

当Spring遇到这个<bean>元素时,它会创建一个CDPlayer实例。<constructor-arg>元素会告知Spring要将一个ID为compactDisc的bean引用传递到CDPlayer的构造器中。

作为替代方案,也可以使用Spring的c-命名空间。以下是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”

xmlns:c=“http://www.springframework.org/schema/c”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd">

在c-命名空间和模式声明之后,我们就可以使用它来声明构造器参数了,如下所示:

c的属性的说明如下:

img

spring的c标签的构造器属性

属性名以“c:”开头,也就是命名空间的前缀。接下来就是要装配的构造器参数名,在此之后是“-ref”,这是一个命名的约定,它会告诉Spring,正在装配的是一个bean的引用,这个bean的名字是compactDisc,而不是字面量“compactDisc”。

还可以将其中引用参数的名称替换成位置信息:

其中“0”代表参数的索引。

下面我们看一下如何将字面量如何装配到对象中去。

我们先新建一个CompactDisc的实现:

package com.springinaction;

import java.ut
il.List;

public class BlankDisc implements CompactDisc {

private String title;

private String artist;

public BlankDisc(String title, String artist){

this.title = title;

this.artist = artist;

}

@Override

public void play() {

System.out.println("Playing " + title + " by " + artist);

}

}

这个实现类中,cd名称和艺术家是可以从构造器注入的。比SgtPeppers的硬编码要灵活。现在我们将已有的SgtPeppers替换为这个类:

还可以替换为c标签写法:

也可以用索引:

通常<constructor-arg>和c-命名空间的功能是相同的,但一种情况是<constructor-arg>能实现,但c-却做不到。

现在我们将BlankDisc加入多磁道的属性,并在构造器中能注入。

package com.springinaction;

import java.util.List;

public class BlankDisc implements CompactDisc {

private String title;

private String artist;

private List tracks;

public BlankDisc(String title, String artist, List tracks ){

this.title = title;

this.artist = artist;

this.tracks = tracks;

}

@Override

public void play() {

System.out.println("Playing " + title + " by " + artist);

for (String track : tracks){

System.out.println("-Track: " + track);

}

}

}

最简单的方式是将列表设置为null

更好的解决方法是提供一个磁道名称的列表。

方案一:

Sgt. Pepper’s Lonely Hearts Club Band

With a Little Help from My Friends

Lucy in the Sky with Diamonds

Getting Better

Fixing a Hole

方案二:

假如构造器的参数还有List,那就使用元素代替。

如果构造器的参数类型是Set,可以把替换为

除了使用构造器注入,我们还可以使用属性的Setter方法进行注入。

假设注入的CDPlayer如下所示

package com.springinaction;

public class CDPlayer implements MediaPlayer {

private CompactDisc compactDisc;

public void setCompactDisc(CompactDisc compactDisc){

this.compactDisc = compactDisc;

}

@Override

public void play() {

compactDisc.play();

}

}

Spring的XML注入如下:

如果运行测试的话,它应该就能通过了。

还有用p标签的方式替代,需要先声明:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xmlns:p=“http://www.springframework.org/schema/p”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd">

使用p-命名空间,装配如下:

p-命名空间的描述如下:

img

spring的p便签的setter方式注入

介绍完bean注入,接下来介绍字面量注入属性

重新修改BlankDisc,使其所有属性都有setter方法

package com.springinaction;

import java.util.List;

public class BlankDisc implements CompactDisc {

private String title;

private String artist;

private List tracks;

public void setTitle(String title){

this.title = title;

}

public void setArtist(String artist){

this.artist = artist;

}

public void setTracks(List tracks){

this.tracks = tracks;

}

@Override

public void play() {

System.out.println("Playing " + title + " by " + artist);

for (String track : tracks){

System.out.println("-Track: " + track);

}

}

}

实现SpringXML的属性注入

Sgt. Pepper’s Lonely Hearts Club Band

With a Little Help from My Friends

Lucy in the Sky with Diamonds

Getting Better

Fixing a Hole

也可以用p-命名空间来替代:

<bean id=“compactDisc” class=“com.springinaction.BlankDisc” p:title=“Sgt. Pepper’s Lonely Hearts Club Band”

p:artist=“The Beatles” p:tracks-ref=“trackList”/>

<util:list id=“trackList”>

Sgt. Pepper’s Lonely Hearts Club Band

With a Little Help from My Friends

Lucy in the Sky with Diamonds

Getting Better

Fixing a Hole

</util:list>

如果想使用util-命名空间,则需要以下声明:

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xmlns:p=“http://www.springframework.org/schema/p”

xmlns:util=“http://www.springframework.org/schema/util”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd

http://www.springframework.org/schema/util

http://www.springframework.org/schema/util/spring-util.xsd">

2.5 导入和混合配置


想在JavaConfig中引用XML配置或者导入其他配置类,如下所示:

package com.springinaction;

import org.springframework.context.annotation.*;

@Configuration

@Import(CDConfig.class) // 导入单个配置类

@Import({CDConfig.class, CDPlayingConfig.class}) // 导入多个配置类

@ImportResource(“classpath:soundsystem.xml”) // 导入xml文件

public class CDPlayerConfig {

}

想在XML配置中引用JavaConfig, 如下所示

sgtPeppers在JavaConfig配置,CDPlayer继续在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”

xmlns:c=“http://www.springframework.org/schema/c”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd">

如果把XML的一部分配置独立写到一个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”

xmlns:c=“http://www.springframework.org/schema/c”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans.xsd">

2.6 小结


Spring框架的核心是Spring容器。容器负责管理应用中组件的生命周期,它会创建这些组件并保证它们的依赖能够得到满足,这样的话,组件才能完成预定的任务。

在本章中,我们看到了在Spring中装配bean的三种主要方式:自动化配置、基于Java的显式配置以及基于XML的显式配置。不管你采用什么方式,这些技术都描述了Spring应用中的组件以及这些组件之间的关系。
我同时建议尽可能使用自动化配置,以避免显式配置所带来的维护成本。但是,如果你确实需要显式配置Spring的话,应该优先选择基于Java的配置,它比基于XML的配置更加强大、类型安全并且易于重构。在本书中的例子中,当决定如何装配组件时,我都会遵循这样的指导意见。
因为依赖注入是Spring中非常重要的组成部分,所以本章中介绍的技术在本书中所有的地方都会用到。基于这些基础知识,下一章将会介绍一些更为高级的bean装配技术,这些技术能够让你更加充分地发挥Spring容器的威力。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值