美文网首页java高级Java学习笔记程序员
Spring依赖注入(控制反转)DI(IOC)的常用方式(构造器

Spring依赖注入(控制反转)DI(IOC)的常用方式(构造器

作者: 重山杨 | 来源:发表于2017-02-11 23:01 被阅读637次

    平常的java开发中,程序员在某个类中需要依赖其它类的方法,通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程序员实例化,而是通过spring容器帮我们new指定实例并且将实例注入到需要该对象的类中。依赖注入的另一种说法是“控制反转”,通俗的理解是:平常我们new一个实例,这个实例的控制权是我们程序员,而控制反转是指new实例工作不由我们程序员来做而是交给spring容器来做。

    通常使用setter方法和构造器方式来进行注入,在过去的开发过程中,这两种注入方式都是非常常用的。spring也同时支持这两种依赖注入的方式:设值注入和构造注入。这两种依赖注入的方式并没有绝对的好坏,知识适应的场景不一样。

    相比而言setter注入具有以下优点:

    1)与传统的JavaBean的写法更相似,程序开发人员更容易理解、接受。通过setter方法设定依赖关系显得更加直观、自然。
    2)对于复杂的依赖关系,如果采用构造注入,会导致构造器过于臃肿,难以阅读。Spring在创建Bean实例时,需要同时实例化其依赖的全部实例,因而导致性能下降。而是用设置注入可以避免这些问题。
    3)尤其在某些属性可选的情况下,多参数的构造器更加笨重。
    
    

    某些情况下,构造注入的优势:

    
    1)构造注入可以再构造器中决定依赖关系的注入顺序,有限依赖的优先注入。例如,组件中其它依赖关系的注入,常常需要依赖于Datasource的注入。采用构造注入,可以在代码中清晰地决定注入顺序。
    2)对于依赖关系无需变化的Bean,构造注入更加有用。因为没有setter方法,所有的依赖关系全部在构造器内设定。因此,无需担心后续代码对依赖关系的破坏。
    3)依赖关系只能在构造器中设定,则只有组建的创建者才能改变组建的依赖关系。队组建的调用者而言,组件内部的依赖关系完全透明,更符合高内聚的原则。
    

    建议:采用设置注入为主,构造注入为辅的注入策略。对于依赖关系无需变化的注入,尽量采用构造注入;而其它的依赖关系的注入,则考虑设值注入。


    一、构造器注入

    这种方式的注入是指带有参数的构造函数注入,再定义属性类时,需要定义带有参数的构造器,属性值通过这一构造器注入到对象中

    相对于使用setter方式进行注入,使用构造器注入时,明确了哪些属性是必须的,通过构造强制依赖关系,不可能实例化不完全的或无法使用的bean。

    构造器注入主要有两种方式:

    • 使用<constructor-arg>元素

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

    使用<construtor-arg>元素进行构造器注入时会使得xml配置文件相对繁琐,但有时能比使用c命名空间进行注入具有更多功能

    1、使用<constructor-arg>元素进行注入

    属性类:

    public class SpringAction {
        //注入对象springDao
        private String name;
        private int salary;
        private User user;
        
        //此处必须提供含参数的构造函数用于注入相关属性
        public SpringAction(String name,int salary,User user){
            this.name = name;
            this.salary = salary;
            this.user = user;
            System.out.println("构造方法调用属性");
        }
            
            public void save(){
            ...
        }
    }
    

    配置文件beans.xml

    <!--配置bean,配置后该类由spring管理-->
    <bean name="user" class="com.bless.springdemo.vo.User"></bean>
    
    --很久参数索引赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <constructor-arg index="0" value="刘晓刚" />
         <constructor-arg index="1" value="3500" />
         <constructor-arg index="2" ref="user"/>
    </bean>
    
    --根据参数类型赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <constructor-arg type="Java.lang.String" value="刘晓刚" />
         <constructor-arg type="java.lang.Intager" value="3500" />
         <constructor-arg type="com.bless.springdemo.vo.User" ref="user"/>
    </bean>
    
    --根据参数名称赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <constructor-arg name="name" value="刘晓刚" />
         <constructor-arg name="salary" value="3500" />
         <constructor-arg name="user" ref="user"/>
    </bean>
    
    --按照参数顺序直接赋值(value)--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <constructor-arg value="刘晓刚" />
         <constructor-arg value="3500" />
         <constructor-arg ref="user"/>
    </bean>
    

    2、使用c命名空间注入

    spring3.0版本后添加的注入方式,简化了xml配置文件内容

    配置文件beans.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:p="http://www.springframework.org/schema/c"       ////使用命名空间时,注意头文件这里要多出这一行
        xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
    
    <!--配置bean,配置后该类由spring管理-->
    <bean name="user" class="com.bless.springdemo.vo.User"></bean>
    
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" 
          c:name-value="小明" 
          c:salary-value="2300"
          c:user-ref="user"/>
    
    --也可以这样使用ref属性--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" 
          c:_-ref="user"/>    
    
    --当只使用value属性时可以这样--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" 
          c:_value="小明" 
          c:_salary="2300"/>
          
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" 
          c:_0="小明" 
          c:_1="2300"/>
          
    </beans>
    

    二、setter注入

    1、常量和bean注入(使用<property>标签)

    属性类:

    public class SpringAction {
        //注入对象springDao
        private String name;
        private int salary;
        private User user;
        
        //此处一定要有属性的setter方法
        public void setName(String name) { 
            this.name = name; 
        } 
        
        public void setSalary(Salary salary) { 
            this.salary = salary; 
        } 
        
        public void setUser(User user) { 
            this.user = user; 
        } 
            
            public void save(){
            ...
        }
    }
    

    配置文件beans.xml

    <!--配置bean,配置后该类由spring管理-->
    <bean name="user" class="com.bless.springdemo.vo.User"></bean>
    
    --很久参数索引赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property index="0" value="jane"/>
         <property index="1" value="3500" />
         <property index="2" ref="user"/>
    </bean>
    
    --根据参数类型赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property type="Java.lang.String" value="刘晓刚" />
         <property type="java.lang.Intager" value="3500" />
         <property type="com.bless.springdemo.vo.User" ref="user"/>
    </bean>
    
    --根据参数名称赋值--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property name="name" value="刘晓刚" />
         <property name="salary" value="3500" />
         <property name="user" ref="user"/>
    </bean>
    
    --按照参数顺序直接赋值(value)--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property value="刘晓刚" />
         <property value="3500" />
         <property ref="user"/>
    </bean>
    

    2、集合对象注入

    配置文件beans.xml

    <!--配置bean,配置后该类由spring管理-->
    <bean name="user" class="com.bless.springdemo.vo.User"></bean>
    
    --注入list参数--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property index="0">
            <list>
                <value>小红</value>
                <value>小明</value>
                <value>小刚</value>
            </list>
         </property>
    </bean>
    
    --在list中引用bean--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property index="0">
            <list>
                <ref bean="user"/>
                <ref bean="student"/>
            </list>
         </property>
    </bean>
    
    --注入map参数--
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
         <property name="map">
            <map>
                <entry key="name" value="小明"/>
                <entry key="name" value="小红"/>
                <entry key="name" value="小刚"/>
            </map>
         </property>
    </bean>
    
    --null注入--
    <property name="wife"><null/></property>
    

    3、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-3.0.xsd">
    
    <!--配置bean,配置后该类由spring管理-->
    <bean name="user" class="com.bless.springdemo.vo.User"></bean>
    
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" 
          p:name="小明" 
          c:salary="2300"
          c:user-ref="user"/>
          
    </beans>
    

    4、使用util命名空间进行集合注入(了解,p命名空间不能注入集合)

    三、工厂方法注入

    1、静态工厂的方法注入

    静态工厂顾名思义,就是通过调用静态工厂的方法来获取自己需要的对象,为了让spring管理所有对象,我们不能直接通过"工程类.静态方法()"来获取对象,而是依然通过spring注入的形式获取:

    package com.bless.springdemo.factory; 
    
    import com.bless.springdemo.dao.FactoryDao; 
    import com.bless.springdemo.dao.impl.FactoryDaoImpl; 
    import com.bless.springdemo.dao.impl.StaticFacotryDaoImpl; 
    
    public class DaoFactory { 
        //静态工厂 
        public static final FactoryDao getStaticFactoryDaoImpl(){ 
            return new StaticFacotryDaoImpl(); 
        } 
    }
    

    同样看关键类,这里我需要注入一个FactoryDao对象,这里看起来跟第一种注入一模一样,但是看随后的xml会发现有很大差别

    public class SpringAction { 
        //注入对象 
        private FactoryDao staticFactoryDao; 
    
        public void staticFactoryOk(){ 
            staticFactoryDao.saveFactory(); 
        } 
        //注入对象的set方法 
        public void setStaticFactoryDao(FactoryDao staticFactoryDao) { 
            this.staticFactoryDao = staticFactoryDao; 
        } 
    } 
    

    Spring的IOC配置文件,注意看<bean name="staticFactoryDao">指向的class并不是FactoryDao的实现类,而是指向静态工厂DaoFactory,并且配置 factory-method="getStaticFactoryDaoImpl"指定调用哪个工厂方法:

    <!--配置bean,配置后该类由spring管理--> 
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction" > 
        <!--(3)使用静态工厂的方法注入对象,对应下面的配置文件(3)--> 
        <property name="staticFactoryDao" ref="staticFactoryDao"></property> 
        </property> 
    </bean> 
    <!--(3)此处获取对象的方式是从工厂类中获取静态方法--> 
    <bean name="staticFactoryDao" class="com.bless.springdemo.factory.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean> 
    

    4.实例工厂的方法注入

    实例工厂的意思是获取对象实例的方法不是静态的,所以你需要首先new工厂类,再调用普通的实例方法:

    public class DaoFactory { 
        //实例工厂 
        public FactoryDao getFactoryDaoImpl(){ 
            return new FactoryDaoImpl(); 
        } 
    }
    

    那么下面这个类没什么说的,跟前面也很相似,但是我们需要通过实例工厂类创建FactoryDao对象:

    public class SpringAction { 
        //注入对象 
        private FactoryDao factoryDao; 
        
        public void factoryOk(){ 
            factoryDao.saveFactory(); 
        } 
        
        public void setFactoryDao(FactoryDao factoryDao) { 
            this.factoryDao = factoryDao; 
        } 
    } 
    

    最后看spring配置文件:

    <!--配置bean,配置后该类由spring管理--> 
    <bean name="springAction" class="com.bless.springdemo.action.SpringAction"> 
        <!--(4)使用实例工厂的方法注入对象,对应下面的配置文件(4)--> 
        <property name="factoryDao" ref="factoryDao"></property> 
    </bean> 
    
    <!--(4)此处获取对象的方式是从工厂类中获取实例方法--> 
    <bean name="daoFactory" class="com.bless.springdemo.factory.DaoFactory"></bean> 
    <bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"></bean> 
    

    四、另外注意:

    通过Spring创建的对象默认是单例的,如果需要创建多实例对象可以在<bean>标签后面添加一个属性:

    <bean name="..." class="..." scope="prototype">  
    

    相关文章

      网友评论

        本文标题:Spring依赖注入(控制反转)DI(IOC)的常用方式(构造器

        本文链接:https://www.haomeiwen.com/subject/ulgoittx.html