java高级Java学习笔记程序员

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

2017-02-11  本文已影响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。

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

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

猜你喜欢

热点阅读