依赖注入
依赖注入
Spring支持两种依赖注入方式,分别是属性注入和构造函数注入.除此之外,Spring 还支持工厂方法注入方式.
属性注入
属性注入是指通过 setXxx() 方式注入Bean 的属性值或依赖对象.由于属性注入方式具有可选择性和灵活性高的有点,因此属性注入是实际应用中最常采用的注入方式.
属性注入要求Bean 提供一个默认的构造函数,并为需要注入的属性提供对应的Setter方法.Spring 先调用 Bean 的默认构造函数实例化Bean 对象,然后通过反射的方式调用 Setter 方法注入属性值.
<bean id="fileManagerConfig" class="cn.com.servyou.xqy.framework.file.FileManagerConfig">
<property name="fileStore" value="sjssjjs"/>
<property name="ftpNginxServer" value="cdknckdsjc"/>
<property name="staticUrl" value="djcjdskcnkd"/>
</bean>
上述代码配置了一个Bean,并为该Bean 的3个属性提供了属性值.具体来说,Bean 的每一个属性对应一个 <property>标签,name 为属性的名称,在Bean 实现类中拥有与其对应的 Setter方法.
public class FileManagerConfig {
private String fileStore;
private String staticUrl;
private String ftpNginxServer;
public FileManagerConfig() {
}
public String getFileStore() {
return this.fileStore;
}
public void setFileStore(String fileStore) {
this.fileStore = fileStore;
}
public String getStaticUrl() {
return this.staticUrl;
}
public void setStaticUrl(String staticUrl) {
this.staticUrl = staticUrl;
}
public String getFtpNginxServer() {
return this.ftpNginxServer;
}
public void setFtpNginxServer(String ftpNginxServer) {
this.ftpNginxServer = ftpNginxServer;
}
}
需要注意的是 Spring 只会检查 Bean 中是否有对应的 Setter 方法,至于Bean 中是否有对应的属性成员变更则不做要求.举个例子,配置文件中的 fileStore 属性配置项仅要求 FileManagerConfig 拥有 setFileStore 方法,但 该 Bean 不一定要拥有 整个成员变量.
JavaBean 关于属性命名的特殊规范:
一般情况下,Java 的属性变量名都是以小写字母开头,如 maxSpeed,branch等,但也存在特殊的情况.如 USA,XML,JavaBean 也允许 以大写字母开头的属性变量名,不过必须满足变量的前两个字母要么全部大些,要么全部小写,如 branch IDCode,IC,ICCard就是合法的,而 iC iCcard,iDCode就是非法的.
构造函数注入
构造函数注入是除属性注入之外另一种常用的注入方式,它保证一些必要的属性在 Bean 实例化时就得到设置,确保Bean 在实例化后就可以使用.
按照类型匹配入参:
<bean id="fileManagerConfig" class="cn.com.servyou.xqy.framework.file.FileManagerConfig">
<constructor-arg type="java.lang.String">
<value>jsjsj</value>
<constructor-arg/>
</bean>
按照索引匹配入参
<bean id="fileManagerConfig" class="cn.com.servyou.xqy.framework.file.FileManagerConfig">
<constructor-arg index="0" value="dkdk"/>
</bean>
联合使用索引和类型
<bean id="fileManagerConfig" class="cn.com.servyou.xqy.framework.file.FileManagerConfig">
<constructor-arg index="0" type="java.lang.String">
<value>jsjsj</value>
<constructor-arg/>
</bean>
通过自身类型反射匹配入参
<bean id="sqlSessionTemplate"
<constructor-arg>
<ref bean="sqlSessionFactory"/>
<constructor-arg>
注意循环依赖问题 构造函数注入是没有办法解决的,只有 属性注入方式才可以解决.
工厂方法注入
非静态工厂方法
<bean id="cacheAdminFactory" lazy-init="false"
class="cn.com.servyou.xqy.framework.cacheframework.cache.CacheAdminFactory">
<!-- 缓存客户端使用 -->
<bean id="cacheClient" factory-bean="cacheAdminFactory" factory-method="getCacheClient"/>
<!-- Redis缓存客户端 -->
<bean id="redisCacheClient" factory-bean="cacheAdminFactory"
factory-method="getRedisCacheClient"/>
CacheAdminFactory 源码
/**
* 由于是系统启动时候,只会初始化一个client对象,所以synchronized可加加不加
*
* @return
*/
public synchronized ICacheClient getCacheClient() {
logger.warn("++++++++++cacheClient init++++++++++");
if (cacheClient == null) {
cacheClient = new ShardedJedisClient(prefix);
}
return cacheClient;
}
/**
* 获取Redis缓存客户端,该接口相对于通用缓存客户端ICacheClient,提供了Redis一些特有的接口。
* <p/>
* 如果要获取通用缓存客户端ICacheClient,则调用getCacheClient(),clientType为redis
*
* @return Redis缓存客户端
*/
public synchronized IRedisCacheClient getRedisCacheClient() {
logger.info("++++++++++RedisCacheClient init++++++++++");
if (redisCacheClient == null) {
redisCacheClient = new ShardedJedisClient(prefix);
}
logger.info("Get Redis CacheClient.");
return redisCacheClient;
}
静态工厂方法:
<bean id="beanId" class="XXX.xxx.xx" factory-method="">