spring几种依赖注入方式以及ref-local/bean,factory-bean,factory-method区别联系

时间:2023-03-09 19:05:28
spring几种依赖注入方式以及ref-local/bean,factory-bean,factory-method区别联系

平常的java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程序员实例化,而是通过spring容器帮我们new指定实例并且将实例注入到需要该对象的类中。

  依赖注入的另一种说法是“控制反转”,通俗的理解是:平常我们new一个实例,这个实例的控制权是我们程序员,而控制反转是指new实例工作不由我们程序员来做而是交给spring容器来做。

spring有多种依赖注入的形式,下面仅介绍spring通过xml进行IOC配置的方式:

1 set注入

这是最简单的注入方式,假设有一个SpringAction,类中需要实例化一个SpringDao对象,那么就可以定义一个private的SpringDao成员变量,然后创建SpringDao的set方法(这是ioc的注入入口):

  1. <span style="font-size: 14px;">package com.bless.springdemo.action;
  2. public class SpringAction {
  3. //注入对象springDao
  4. private SpringDao springDao;
  5. //一定要写被注入对象的set方法
  6. public void setSpringDao(SpringDao springDao) {
  7. this.springDao = springDao;
  8. }
  9. public void ok(){
  10. springDao.ok();
  11. }
  12. }</span>
<span style="font-size: 14px;">package com.bless.springdemo.action;
public class SpringAction {
//注入对象springDao
private SpringDao springDao;
//一定要写被注入对象的set方法
public void setSpringDao(SpringDao springDao) {
this.springDao = springDao;
}
public void ok(){
springDao.ok();
}
}</span>
    随后编写spring的xml文件,<bean>中的name属性是class属性的一个别名,class属性指类的全名,因为在SpringAction中有一个公共属性Springdao,所以要在<bean>标签中创建一个<property>标签指定SpringDao。<property>标签中的name就是SpringAction类中的SpringDao属性名,ref指下面<bean
name="springDao"...>,这样其实是spring将SpringDaoImpl对象实例化并且调用SpringAction的setSpringDao方法将SpringDao注入:
  1. <!--配置bean,配置后该类由spring管理-->
  2. <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
  3. <!--(1)依赖注入,配置当前类中相应的属性-->
  4. <property name="springDao" ref="springDao"></property>
  5. </bean>
  6. <bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>
<!--配置bean,配置后该类由spring管理-->
<bean name="springAction" class="com.bless.springdemo.action.SpringAction">
<!--(1)依赖注入,配置当前类中相应的属性-->
<property name="springDao" ref="springDao"></property>
</bean>
<bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>
2 构造器注入
    这种方式的注入是指带有参数的构造函数注入,看下面的例子,我创建了两个成员变量SpringDao和User,但是并未设置对象的set方法,所以就不能支持第一种注入方式,这里的注入方式是在SpringAction的构造函数中注入,也就是说在创建SpringAction对象时要将SpringDao和User两个参数值传进来:
  1. public class SpringAction {
  2. //注入对象springDao
  3. private SpringDao springDao;
  4. private User user;
  5. public SpringAction(SpringDao springDao,User user){
  6. this.springDao = springDao;
  7. this.user = user;
  8. System.out.println("构造方法调用springDao和user");
  9. }
  10. public void save(){
  11. user.setName("卡卡");
  12. springDao.save(user);
  13. }
  14. }
public class SpringAction {
//注入对象springDao
private SpringDao springDao;
private User user;
public SpringAction(SpringDao springDao,User user){
this.springDao = springDao;
this.user = user;
System.out.println("构造方法调用springDao和user");
} public void save(){
user.setName("卡卡");
springDao.save(user);
}

}

在XML文件中同样不用<property>的形式,而是使用<constructor-arg>标签,ref属性同样指向其它<bean>标签的name属性:
  1. <!--配置bean,配置后该类由spring管理-->
  2. <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
  3. <!--(2)创建构造器注入,如果主类有带参的构造方法则需添加此配置-->
  4. <constructor-arg ref="springDao"></constructor-arg>
  5. <constructor-arg ref="user"></constructor-arg>
  6. </bean>
  7. <bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>
  8. <bean name="user" class="com.bless.springdemo.vo.User"></bean>
<!--配置bean,配置后该类由spring管理-->
<bean name="springAction" class="com.bless.springdemo.action.SpringAction">
<!--(2)创建构造器注入,如果主类有带参的构造方法则需添加此配置-->
<constructor-arg ref="springDao"></constructor-arg>
<constructor-arg ref="user"></constructor-arg>
</bean>
<bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>
<bean name="user" class="com.bless.springdemo.vo.User"></bean>

解决构造方法参数的不确定性,你可能会遇到构造方法传入的两参数都是同类型的,为了分清哪个该赋对应值,则需要进行一些小处理:

下面是设置index,就是参数位置:
  1. <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
  2. <constructor-arg index="0" ref="springDao"></constructor-arg>
  3. <constructor-arg index="1" ref="user"></constructor-arg>
  4. </bean>
<bean name="springAction" class="com.bless.springdemo.action.SpringAction">
<constructor-arg index="0" ref="springDao"></constructor-arg>
<constructor-arg index="1" ref="user"></constructor-arg>
</bean>

另一种是设置参数类型:

  1. <constructor-arg type="java.lang.String" ref=""/>  <strong>
  2. </strong>
<constructor-arg type="java.lang.String" ref=""/>  <strong>
</strong>
3 静态工厂的方法注入
   静态工厂顾名思义,就是通过调用静态工厂的方法来获取自己需要的对象,为了让spring管理所有对象,我们不能直接通过"工程类.静态方法()"来获取对象,而是依然通过spring注入的形式获取:
  1. <span style="font-size:14px;">package com.bless.springdemo.factory;
  2. import com.bless.springdemo.dao.FactoryDao;
  3. import com.bless.springdemo.dao.impl.FactoryDaoImpl;
  4. import com.bless.springdemo.dao.impl.StaticFacotryDaoImpl;
  5. public class DaoFactory {
  6. //静态工厂
  7. public static final FactoryDao getStaticFactoryDaoImpl(){
  8. return new StaticFacotryDaoImpl();
  9. }
  10. } </span><span style="font-size:12px;"> </span>
<span style="font-size:14px;">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();

}

} </span><span style="font-size:12px;"> </span>

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

  1.  public class SpringAction {    
  2.         //注入对象    
  3.     private FactoryDao staticFactoryDao;    
  4.         
  5.     public void staticFactoryOk(){    
  6.         staticFactoryDao.saveFactory();    
  7.     }    
  8.     //注入对象的set方法    
  9.     public void setStaticFactoryDao(FactoryDao staticFactoryDao) {    
  10.         this.staticFactoryDao = staticFactoryDao;    
  11.     }    
  12. }    
  13.    
 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"指定调用哪个工厂方法:
  1. <!--配置bean,配置后该类由spring管理-->
  2. <bean name="springAction" class="com.bless.springdemo.action.SpringAction" >
  3. <!--(3)使用静态工厂的方法注入对象,对应下面的配置文件(3)-->
  4. <property name="staticFactoryDao" ref="staticFactoryDao"></property>
  5. </bean>
  6. <!--(3)此处获取对象的方式是从工厂类中获取静态方法-->
  7. <bean name="staticFactoryDao" class="com.bless.springdemo.factory.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean>
<!--配置bean,配置后该类由spring管理-->
<bean name="springAction" class="com.bless.springdemo.action.SpringAction" >
<!--(3)使用静态工厂的方法注入对象,对应下面的配置文件(3)-->
<property name="staticFactoryDao" ref="staticFactoryDao"></property>
</bean>
<!--(3)此处获取对象的方式是从工厂类中获取静态方法-->
<bean name="staticFactoryDao" class="com.bless.springdemo.factory.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean>
4 实例工厂的方法注入
实例工厂的意思是获取对象实例的方法不是静态的,所以你需要首先new工厂类,再调用普通的实例方法:
  1. public class DaoFactory {
  2. //实例工厂
  3. public FactoryDao getFactoryDaoImpl(){
  4. return new FactoryDaoImpl();
  5. }
  6. }
public class DaoFactory {
//实例工厂
public FactoryDao getFactoryDaoImpl(){
return new FactoryDaoImpl();
}
}
那么下面这个类没什么说的,跟前面也很相似,但是我们需要通过实例工厂类创建FactoryDao对象:
  1. public class SpringAction {
  2. //注入对象
  3. private FactoryDao factoryDao;
  4. public void factoryOk(){
  5. factoryDao.saveFactory();
  6. }
  7. public void setFactoryDao(FactoryDao factoryDao) {
  8. this.factoryDao = factoryDao;
  9. }
  10. }
public class SpringAction {
//注入对象
private FactoryDao factoryDao;
public void factoryOk(){
factoryDao.saveFactory();
}
public void setFactoryDao(FactoryDao factoryDao) {
this.factoryDao = factoryDao;
}

}

最后看spring配置文件:
  1. <!--配置bean,配置后该类由spring管理-->
  2. <bean name="springAction" class="com.bless.springdemo.action.SpringAction">
  3. <!--(4)使用实例工厂的方法注入对象,对应下面的配置文件(4)-->
  4. <property name="factoryDao" ref="factoryDao"></property>
  5. </bean>
  6. <!--(4)此处获取对象的方式是从工厂类中获取实例方法-->
  7. <bean name="daoFactory" class="com.bless.springdemo.factory.DaoFactory"></bean>
  8. <bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"></bean>
<!--配置bean,配置后该类由spring管理-->
<bean name="springAction" class="com.bless.springdemo.action.SpringAction">
<!--(4)使用实例工厂的方法注入对象,对应下面的配置文件(4)-->
<property name="factoryDao" ref="factoryDao"></property>
</bean>
&lt;!--(4)此处获取对象的方式是从工厂类中获取实例方法--&gt;
&lt;bean name="daoFactory" class="com.bless.springdemo.factory.DaoFactory"&gt;&lt;/bean&gt;
&lt;bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"&gt;&lt;/bean&gt; </pre></div>
总结
Spring IOC注入方式用得最多的是(1)(2)种,多谢多练就会非常熟练。
        另外注意:通过Spring创建的对象默认是单例的,如果需要创建多实例对象可以在<bean>标签后面添加一个属性:
  1. <bean name="..." class="..." scope="prototype">
<bean name="..." class="..." scope="prototype">  

--------------------------------------------------------------完--------------------------------------------------------------------

附录:

一 ref local和ref bean

<ref local="xx"/>

  用"local"属性指定目标其实是指向同一文件内对应"id"属性值为此"local"值的索引

"local"属性的值必须和目标bean的id属性相同。如果同一文件内没有匹配的元素,xml解析器将提示错误。同样,如果目标在同一XML文件内,使用"local"变量是最好的选择(为了尽可能早地知道错误)

<ref bean="xx"/>

  用"bean"属性指定目标bean是最常规的形式,这允许创建索引到任何同一个容器内的bean(无论是否在同一XML 文件中)或者父级的容器内的bean。"bean"属性的值可以和目标bean的"id"属性相同,也可以和目标bean的"name"属性内的一个值相同

 可以这么说, <ref bean=""/> 是寻找全局中的 bean; <ref local=""/> 是寻找本 xml 文件中的 bean

 

<ref> 提供了如下几方面的属性 :

     1)bean: 在当前 Spring XML 配置文件中,或者在同一 BeanFactory(ApplicationContext) 中的其他 JavaBean 中 .

2)local: 在当前 Spring XML 配置文件中 . 其依赖的 JavaBean 必须存在于当前 Spring XML 配置 文件中 . 如果借助于 Spring IDE, 则在编译期可以对其依赖的 JavaBean 进行验证。基于 local 方式,开发者能够使用到 XML 本身提供 的优势,而进行验证。

3)parent: 用于指定其依赖的父 JavaBean 定义。

二  xml配置文件中的bean的三种方法

<bean id="itemf" class="cn.itcareers.m03.ItemFactory" /> 是个普通的Bean;

<bean id="item" factory-bean="itemf" factory-method="getItem" />是工厂Bean。

工厂Bean返回的不是一个实例,而是由工厂getItem方法返回的对象。

2.1 最常见,也是缺省,是调用spring的缺省工厂类

spring缺省工厂类:org.springframework.beans.factory.support.DefaultListableBeanFactory

使用其静态方法preInstantiateSingletons()

配置文件:最普通最基本的定义一个普通bean

  1. <bean id="DvdTypeDAOBean" class="com.machome.dvd.impl.DvdTypeDAO" > </bean>
        <bean id="DvdTypeDAOBean" class="com.machome.dvd.impl.DvdTypeDAO" > </bean>

看日志:

2010-5-29 0:48:48 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons

2.2 使用用户自定义的工厂类的静态方法进行创建

定义工厂类,返回new实例

  1. public class DvdTypeDAOFactory {
  2. public static DvdTypeDAO getInstance(){
  3. return new DvdTypeDAO();
  4. }
  5. }
public class DvdTypeDAOFactory {
public static DvdTypeDAO getInstance(){
return new DvdTypeDAO();
}
}

这里只是简单的定义了一个工厂类,你可以定义更复杂的:

比如单例类(提高利用效率),

比如含有ThreadLocal属性的类(多线程下解决线程安全问题)。

另外:spring配置文件中, bean的class不需要new的类,而是设成工厂类,然后后面加factory-method属性

  1. <bean id="DvdTypeDAOBean" class="com.machome.core.impl.DvdTypeDAOFactory"  factory-method="getInstance" />
<bean id="DvdTypeDAOBean" class="com.machome.core.impl.DvdTypeDAOFactory"  factory-method="getInstance" />  

3.3 在spring中建立工厂类的Bean,然后调用该工厂bean建立真正的bean

  1. public class DvdTypeDAOFactory {
  2. public DvdTypeDAO getInstance(){
  3. return new DvdTypeDAO();
  4. }
  5. }
public class DvdTypeDAOFactory {
public DvdTypeDAO getInstance(){
return new DvdTypeDAO();
}
}

spring配置文件中

  1. <bean id="DvdTypeDAOFactoryBean" class="com.machome.core.impl.DvdTypeDAOFactory" />
 <bean id="DvdTypeDAOFactoryBean" class="com.machome.core.impl.DvdTypeDAOFactory" />

这个工厂bean还是被spring的DefaultListableBeanFactory. preInstantiateSingletons创建

  1. <bean id="DvdTypeDAOBean" factory-bean="DvdTypeDAOFactoryBean"   factory-method="getInstance" />
<bean id="DvdTypeDAOBean" factory-bean="DvdTypeDAOFactoryBean"   factory-method="getInstance" />  
        </div>
posted @
2018-05-22 14:56 
星朝 
阅读(...) 
评论(...) 
编辑 
收藏