Spring实战之XML与JavaConfig的混合配置详解
作者:杰锅锅 发布时间:2023-07-11 18:07:20
前言
之前提到了关于Spring的显示配置方式有两种,一种是基于XML配置,一种是基于JavaConfig的方式配置。对于这两种配置方式并不是互斥关系,相反,他们两能够相互融合,有效的搭配完成Spring的bean注入。
这里分别介绍如何在JavaConfig中引用XML配置的bean以及如何在XML配置中引用JavaConfig。下面话不多说,来一起看看详细的介绍吧。
一、JavaConfig中引用XML配置的bean
上篇我们看到配置类CDPlayerConfig具体代码如下
@Configuration
public class CDPlayerConfig {
@Bean
public CompactDisc compactDisc() {
return new SgtPeppers();
}
@Bean
public CDPlayer cdPlayer(CompactDisc compactDisc) {
return new CDPlayer(compactDisc);
}
}
在该配置类中注册了两个bean。假设让CDPlayerConfig现在矫情一把,它认为自己内部注册的bean太多了,它有点管不过来了(确实有点矫情过了),它准备把compactDisc这个bean踢出去,让别人来管管。
让谁管呢,这时候我们又新建了另外一个配置类CDConfig
@Configuration
public class CDConfig {
@Bean
public CompactDisc compactDisc() {
return new SgtPeppers();
}
}
这个配置类简洁到不行,只有一个CompactDisc的bean。
现在有了这两个配置类,也集齐了两个bean,那么该如何把他们整合到一起呢,建立起它们的联系。这时候我们新建一个连接类SoundSystemConfig
@Configuration
@Import(CDPlayerConfig.class, CDConfig.class)
public class SoundSystemConfig {
}
这样就解决了在一个配置类中要加载的bean过多的问题。
那么问题来了,如果这时候CompactDisc接口的实现类BlankDisc是声明在XML中,这两种显示声明又会如何结合呢。
public class BlankDisc implements CompactDisc {
private String title;
private String artist;
private List<String> tracks;
public BlankDisc(String title, String artist, List<String> tracks) {
this.title = title;
this.artist = artist;
this.tracks = tracks;
}
public void play() {
System.out.println("Playing " + title + " by " + artist);
for (String track : tracks) {
System.out.println("-Track: " + track);
}
}
}
具体该类在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">
<bean id="compactDisc"
class="soundsystem.BlankDisc"
c:_0="Sgt. Pepper's Lonely Hearts Club Band"
c:_1="The Beatles">
<constructor-arg>
<list>
<value>Sgt. Pepper's Lonely Hearts Club Band</value>
<value>With a Little Help from My Friends</value>
<value>Lucy in the Sky with Diamonds</value>
<value>Getting Better</value>
<value>Fixing a Hole</value>
<!-- ...other tracks omitted for brevity... -->
</list>
</constructor-arg>
</bean>
</beans>
注意:这里的c:_0
和c:_1
表示分别要传入到构造函数中的第一个参数和第二个参数的值,写法等价于
<bean id="compactDisc"
class="soundsystem.BlankDisc">
<constructor-arg value="Sgt. Pepper's Lonely Hearts Club Band"/>
<constructor-arg value="The Beatles">
</bean>
或者
<bean id="compactDisc"
class="soundsystem.BlankDisc"
c:_title="Sgt. Pepper's Lonely Hearts Club Band"
c:_artist="The Beatles"/>
现在BlankDisc已经配置在XML中,那么Spring如何去加载这个XML配置文件读取到这个配置信息并加载呢。这里我们可以通过一个注解@ImportResource来完成这个操作。假设上面关于BlankDisc类的配置信息存放在cd-config.xml文件中,该文件在项目的根路径下。通过修改SoundSystemConfig即可完成加载bean。
@Configuration
@Import(CDPlayerConfig.class)
@ImportResource("classpath:cd-config.xml")
public class SoundSystemConfig {
}
如上所示,一个是通过JavaConfig配置的CDPlayer,一个是通过XML声明的BlankDisc,通过如上处理,实现了都能被Spring容器加载。
二、XML配置中引用JavaConfig
CDPlayerConfig彻底的矫情了一把,XML配置文件表示不服,CDPlayerConfig受不了两个bean,那我就能受得了?(好好好,朕特批XML矫情一次)
其实我们都知道,在XML中声明两个bean不是什么大问题,即便受不了,我们还有一个这种方案,XML中也可以使用Import标签实现导入其他的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">
<import resource="cd-config.xml">
<bean id="cdPlayer"
class="soundsystem.CDPlayer"
c:cd-ref="compactDisc" />
</beans>
注意:这里的c:cd-ref="compactDisc"
等价于添加了属性<constructor-arg ref="compactDisc"
但是XML认为这样它还是受不了,它将<import resource="cd-config.xml">
换成了<bean class="soundsystem.CDConfig" />
。这下畅快多了,因为这样就实现了两种配置的结合了,一种是配置在JavaConfig中的BlankDisc,还有一种是在XML中配置的CDPlayer。
测试
分别对于两种方式编写ut测试如下
ImportJavaConfigTest
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:cdplayer-config.xml")
public class ImportJavaConfigTest {
@Rule
public final StandardOutputStreamLog log = new StandardOutputStreamLog();
@Autowired
private MediaPlayer player;
@Test
public void play() {
player.play();
assertEquals(
"Playing Sgt. Pepper's Lonely Hearts Club Band by The Beatles\r\n",
log.getLog());
}
}
ImportXmlConfigTest
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes=SoundSystemConfig.class)
public class ImportXmlConfigTest {
@Rule
public final StandardOutputStreamLog log = new StandardOutputStreamLog();
@Autowired
private MediaPlayer player;
@Test
public void play() {
player.play();
assertEquals(
"Playing Sgt. Pepper's Lonely Hearts Club Band by The Beatles\r\n" +
"-Track: Sgt. Pepper's Lonely Hearts Club Band\n\n" +
"-Track: With a Little Help from My Friends\n\n" +
"-Track: Lucy in the Sky with Diamonds\n\n" +
"-Track: Getting Better\n\n" +
"-Track: Fixing a Hole\n\n",
log.getLog());
}
}
测试结果
至此,我们领略了
•JavaConfig中引用XML配置的bean
•XML配置中引用JavaConfig
这两种显示配置bean如何融洽的配合起来。
来源:http://www.cnblogs.com/bigdataZJ/p/SpringInAction3.html


猜你喜欢
- public class Change{ public static void main(String args[]){ fin
- using System; using System.Drawing; using System.Collec
- 今天在做一个java调用windows的压缩命令时遇到一奇怪问题代码如下:String cmd ="C:/Program File
- 1,现在因为遇到一个读取pdf文件文本信息遇到乱么问题,才找到这个文本字符串的编码转换的实现方式来判断是否存在乱码(0>乱码>2
- Mybatis注解查找@Select( "SELECT * FROM tt_user WHERE username Like #{
- 1.依赖的jar文件 jsch-0.1.53.jar2.登录方式有密码登录,和密匙登录 代码:主函数:import java.ut
- 实例如下:using System;using System.Linq.Expressions;class DynamicPredicate
- 我们知道在C语言编译时,有那么几个常用的优化编译选项,分别是-O0,-O1,-O2,-O3以及-Os。之前一直觉得既然是优化选
- 前言最近测试反馈一个问题,某个查询全量信息的接口,有时候返回全量数据,符合预期,但是偶尔又只返回1条数据,简直就是“见鬼
- 本文实例为大家分享了Android学习笔记之蓝牙功能的具体代码,供大家参考,具体内容如下蓝牙:短距离无线通讯技术标准。蓝牙协议分为4层,即核
- 一、NIOjava.nio全称java non-blocking IO,是指jdk1.4 及以上版本里提供的新api(New IO) ,为所
- 什么是不可变对象?String对象是不可变的,但这仅意味着你无法通过调用它的公有方法来改变它的值。众所周知, 在Java中, String类
- 静态变量初始化顺序1.简单规则首先先看一段最普遍的JAVA代码:public class Test{ public static Test1
- 装箱(boxing)和拆箱(unboxing)是C#类型系统的核心概念.是不同于C与C++的新概念!,通过装箱和拆箱操作,能够在值类型和引用
- 定义强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器宁愿抛出OOM(OutOfMemoryError)也不会回收它。说明不要被
- 一 .概述先讲缓存实现,主要是mybatis一级缓存,二级缓存及缓存使用后续补充Mybatis缓存的实现是基于Map的,从缓存里面读写数据是
- 引言:在Object基类中,有一个方法叫clone,产生一个前期对象的克隆,克隆对象是原对象的拷贝,由于引用类型的存在,有深克隆和浅克隆之分
- Android 中的危险权限详细整理前言:Android 中有上百种权限,现在将所有的权限归为两类:一类是普通权限一类的危险权限普通权限是指
- 生命周期速览优先级servlet 的声明周期由 tomcat 服务器自行管辖,程序员无法插手;只要没有通过 url 访问 servlet,那
- 顺序结构顺序结构就是按照代码从上往下执行,我们运行的程序就是从上往下的顺序结构,当遇到方法的时候,才去执行方法。例如:System.out.