比如实现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()
);
}
}
测试成功
当自动化的方案行不通的时候,就必须采用显示装配的方式。显示装配有两种可选方案: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);
}
需要注意的有两点:
-
构造器中不能用new创建的对象(这个对象的类是已经在Spring中被声明的),因为Spring会拦截所有对已声明对象的调用,并确保直接返回该方法所创建的bean。这是由于Spring所创建的bean都是单例的。
-
compactDisc会在Spring中寻找已经实现CompactDisc的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
下面我们来看一些特征
-
不再需要直接负责创建SgtPeppers的实例,而在基于JavaConfig的配置中需要。当Spring发现这个元素时,它将会调用SgtPeppers的默认构造器来创建bean。
-
这个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的属性的说明如下:
spring的c标签的构造器属性
属性名以“c:”开头,也就是命名空间的前缀。接下来就是要装配的构造器参数名,在此之后是“-ref”,这是一个命名的约定,它会告诉Spring,正在装配的是一个bean的引用,这个bean的名字是compactDisc,而不是字面量“compactDisc”。
还可以将其中引用参数的名称替换成位置信息:
其中“0”代表参数的索引。
下面我们看一下如何将字面量如何装配到对象中去。
我们先新建一个CompactDisc的实现:
package com.springinaction;
import java.util.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-命名空间的描述如下:
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的属性注入
最后
还有Java核心知识点+全套架构师学习资料和视频+一线大厂面试宝典+面试简历模板可以领取+阿里美团网易腾讯小米爱奇艺快手哔哩哔哩面试题+Spring源码合集+Java架构实战电子书+2021年最新大厂面试题。
actDisc {
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的属性注入
最后
还有Java核心知识点+全套架构师学习资料和视频+一线大厂面试宝典+面试简历模板可以领取+阿里美团网易腾讯小米爱奇艺快手哔哩哔哩面试题+Spring源码合集+Java架构实战电子书+2021年最新大厂面试题。
[外链图片转存中…(img-Do5fGiJJ-1714505420196)]