依赖注入

2020-12-29  本文已影响0人  码而优则仕

依赖注入

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="">
上一篇下一篇

猜你喜欢

热点阅读