Spring注入方式有哪些

在Spring配置文件中使用XML文件进行配置,实际上是让Spring执行了相应的代码,例如:

使用<bean>元素,实际上是让Spring执行无参或有参构造器
使用<property>元素,实际上是让Spring执行一次setter方法

但Java程序还可能有其他类型的语句:调用getter方法、调用普通方法、访问类或对象的Field等,而Spring也为这种语句提供了对应的配置语法:

调用getter方法:使用PropertyPathFactoryBean
调用类或对象的Filed值:使用FiledRetrievingFactoryBean
调用普通方法:使用MethodInvokingFactoryBean

注入其他Bean的属性值

PropertyPathFactoryBean用来获得目标Bean的属性值(实际上就是调用getter方法返回的值),获得的值可以注入给其他的Bean,也可以直接定义新的Bean。看如下的配置文件:

<bean id="person" class="com.abc.Person">
 <property name="age" value="30" />
 <property name="son">
  <!-- 使用嵌套Bean定义属性值 -->
  <bean class="com.abc.service.Son">
   <property name="age" value="11" />
  </bean>
 </property>
</bean>

<bean id="son2" class="com.abc.service.Son">
 <!-- age属性不是直接注入,而是将person中的son的age属性赋值给son2的age属性 -->
 <property name="age">
  <!-- 注意这里使用的是PropertyPathFactoryBean -->
  <bean id="person.son.age"
   class="org.springframework.beans.factory.config.PropertyPathFactoryBean" />
 </property>
</bean>

其中Person类和Son类的属性可以从配置文件中看出,这不再给出。主程序如下:

public class Test {
 public static void main(String args[]) {
  ApplicationContext ac =
   new ClassPathXmlApplicationContext("applicationContext.xml");
  System.out.println("age=" + ac.getBean("son2", Son.class).getAge());
 }
}

输出结果:
age=11

Bean实例的属性值,不仅可以注入另一个Bean,还可将Bean实例的属性值直接定义成Bean实例,这也是通过PropertyPathFactoryBean完成的。对上面的配置文件增加这样一段:

<bean id="son1"
 class="org.springframework.beans.factory.config.PropertyPathFactoryBean">
 <!-- 确定目标Bean,表明son1来自哪个Bean的组件 -->
 <property name="targetBeanName" value="person" />
 <!-- 确定属性,表明son1来自目标Bean的哪个属性 -->
 <property name="propertyPath" value="son" />
</bean>

执行上面的Test类,把son2换成son1,结果一样。

注入其他Bean的Field值

通过FieldRetrievingFactoryBean类,可以将其他Bean的Field值注入给其他Bean,或者直接定义新的Bean。下面是配置片段:

<bean id="son" class="com.abc.service.Son">
 <property name="age">
  <bean id="java.sql.connection.TRANSACTION_SERIALIZABLE"
   class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean" />
 </property>
</bean>

测试主程序与上文定义的类似,这里不再提供,执行结果如下:
age=8

在这个配置中,son对象的age的值,等于java.sql.Connection.TRANSACTION_SERIALIZABLE的 值。在上面的定义中,定义FieldRetrievingFactoryBean工厂Bean时,指定的id并不是该Bean实例的唯一标识,而是指定 Field的表达式(即将要被取出来的值)。

注意:Field既可以是静态的,也可以是非静态的。上面的配置片段指定的Field表达式是静态Field值,因此可以通过类名直接访问。如 果Field值是非静态的,则应该通过容器中已经存在的Bean来访问——即Field表达式的第一个短语应该是容器中已经存在的Bean。
Field值也可以定义成Bean实例,例如,在配置文件中增加下面一段:

<bean id="age"
 class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean">
 <!-- targetClass指定Field所在的目标类 -->
 <property name="targetClass" value="java.sql.Connection" />
 <!-- targetField指定Field名 -->
 <property name="targetField" value="TRANSACTION_SERIALIZABLE" />
</bean>

在主程序中增加如下输出:
System.out.println("age=" + ac.getBean("age"));
执行结果和上文一样。

使用FieldRetrievingFactoryBean获取Field值时,必须指定如下两个属性:
targetClass或targetObject:分别用于指定Field值所在的目标累或目标对象。如果需要获得的Field是静态的,则使用targetClass指定目标累;如果Field是非静态的,则使用targetObject指定目标对象
targetField:指定目标类或目标对象的Field名

如果Field是个静态Field,则有一种更加简洁的写法:

<bean id="age"
 class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean">
 <!-- value指定哪个类的哪个静态域值 -->
 <property name="staticField" value="java.sql.Connection.TRANSACTION_SERIALIZABLE" />
</bean>

注入其他Bean的方法返回值

通过MethodInvokingFactoryBean工厂Bean,可将目标方法的返回值注入为Bean的属性值。这个工厂Bean用来获 取指定方法的返回值,该方法既可以是静态方法,也可以是实例方法;这个值既可以被注入到指定Bean实例的指定属性,也可以直接定义成Bean实例。看例 子:

<bean id="valueGenerator" class="com.abc.util.ValueGenerator" />
<bean id="son1" class="com.abc.service.Son">
 <property name="age">
  <!-- 获取方法返回值:调用valueGenerator的getValue方法 -->
  <bean
   class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">
   <property name="targetObject" ref="valueGenerator" />
   <property name="targetMethod" value="getValue" />
  </bean>
 </property>
</bean>

下面是ValueGenerator:

public class ValueGenerator {
 public int getValue() { return 2; }
 public static int getStaticValue () { return 3;}
}

测试程序依旧打印son1中age的值,代码略,结果如下:
age=2

如果要调用静态方法,则把配置修改为:

<bean id="son1" class="com.abc.service.Son">
 <property name="age">
  <!-- 获取方法返回值:调用valueGenerator的getStaticValue方法 -->
  <bean
   class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">
   <property name="targetClass" value="com.abc.util.ValueGenerator" />
   <property name="targetMethod" value="getStaticValue" />
  </bean>
 </property>
</bean>

测试结果为:
age=3

由于Java是支持重载的,只给定方法名,还不足以能够确定调用哪个方法,通过上面的配置能调用成功是因为ValueGenerator中的两个方法都没有参数。如果方法中有参数,该如何配置呢?在配置文件中加入以下内容:

<bean id="sysProps"
 class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">
 <property name="targetClass" value="java.lang.System" />
 <property name="targetMethod" value="getProperties" />
<bean>
<bean id="javaVersion" class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">
 <!-- 指向上面的sysProps Bean -->
 <property name="targetObject" value="sysProps" />
 <property name="targetMethod" value="getProperty" />
 <!-- 这里配置参数 -->
 <property name="arguments">
  <!-- 使用list元素列出调用方法的多个参数 -->
  <list>
   <value>java.version</value>
  </list>
 </property>
<bean>

上例中相当于用”java.version”作为参数调用了java.lang.System的getProperty方法,返回值将创建一个名为javaVersion的Bean。即相当于:
javaVersion = java.lang.System.getProperty("java.version");

和前文中的Field一样,如果要调用的方法为静态方法,也有一种更加简洁的方法:

<bean id="myBean"
 class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">
 <!-- 使用staticMethod属性,直接指定目标类的目标方法 -->
 <property name="staticMethod" value="com.abc.util.ValueGenerator.getStaticValue" />
</bean>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 详析Spring中依赖注入的三种方式

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

  • 深入解析Java的Spring框架中bean的依赖注入

    每一个基于java的应用程序都有一个共同工作来展示给用户看到的内容作为工作的应用几个对象.当编写一个复杂的Java应用程序,应用程序类应该尽可能独立其他Java类来增加重复使用这些类,并独立于其他类别的测试它们,而这样做单元测试的可能性.依赖注入(或有时称为布线)有助于粘合这些类在一起,同时保持他们的独立. 考虑有其中有一个文本编辑器组件的应用程序,要提供拼写检查.标准的代码将看起来像这样: public class TextEditor { private SpellChecker spell

  • 详解Java的Spring框架中bean的注入集合

    使用value属性和使用<property>标签的ref属性在你的bean配置文件中的对象引用,这两种情况下可以处理单值到一个bean,如果你想通过多元值,如Java Collection类型List, Set, Map 及 Properties.要处理这种情况,Spring提供了四种类型的如下集合的配置元素: 可以使用<list> 或<set> 来连接任何实现java.util.Collection或数组. 会遇到两种情况(a)将收集的直接的值及(b)传递一个bean

  • 实例讲解Java的Spring框架中的控制反转和依赖注入

    近来总是接触到 IoC(Inversion of Control,控制反转).DI(Dependency Injection,依赖注入)等编程原则或者模式,而这些是著名 Java 框架 Spring.Struts 等的核心所在.针对此查了 Wikipedia 中各个条目,并从图书馆借来相关书籍,阅读后有些理解,现结合书中的讲解以及自己的加工整理如下: eg1 问题描述: 开发一个能够按照不同要求生成Excel或 PDF 格式的报表的系统,例如日报表.月报表等等.   解决方案: 根据"面向接口编

  • Spring不能注入Static变量的原因及Spring注入静态变量

    下面给大家介绍spring不能注入static变量的原因,具体详情如下所示: Spring 依赖注入 是依赖 set方法 set方法是 是普通的对象方法 static变量是类的属性 @Autowired private static JdbcTemplate jdbcTemplate; 单纯看这个注入过程是没有报错的,但是在接下来的jdbcTemplate.query()会报空指针错误. ps:Spring注入静态变量 今天碰到一个问题,我的一个工具类提供了几种静态方法,静态方法需要另外一个类的

  • Java的Spring框架中bean的继承与内部bean的注入

    bean的定义继承 bean定义可以包含很多的配置信息,包括构造函数的参数,属性值,比如初始化方法,静态工厂方法名等容器的具体信息. 子bean定义从父定义继承配置数据.子的定义可以覆盖一些值,或者根据需要添加其他. Spring bean定义继承无关,与Java类的继承,但继承的概念是一样的.你可以定义一个父bean定义为模板和其他孩子bean可以从父bean继承所需的配置. 当使用基于XML的配置元数据,指明一个子bean定义使用所在的当前属性指定的父bean作为这个属性的值. 例如: 让我

  • Spring中属性注入详解

    本文演示了int.String.数组.list.set.map.Date等属性的注入. 其中Date类型的注入则是借助了Spring提供的属性编辑器来实现的,首先是用到的五个实体类 package com.jadyer.model; import java.util.Date; import java.util.List; import java.util.Map; import java.util.Set; /** * 常见属性的注入 * @see 包括int,String,Array,lis

  • 详解Java Spring各种依赖注入注解的区别

    注解注入顾名思义就是通过注解来实现注入,Spring和注入相关的常见注解有Autowired.Resource.Qualifier.Service.Controller.Repository.Component. Autowired是自动注入,自动从spring的上下文找到合适的bean来注入 Resource用来指定名称注入 Qualifier和Autowired配合使用,指定bean的名称 Service,Controller,Repository分别标记类是Service层类,Contro

  • JavaWeb Spring依赖注入深入学习

    一.依赖注入(DI) 依赖注入听起来很高深的样子,其实白话就是:给属性赋值.一共有两种方法,第一是以构造器参数的形式,另外一种就是以setting方法的形式. 1 构造器注入 1 使用构造器注入 使用xml的注入方式 A. 通过参数的顺序 <constructor-arg index="0"><value>张三</value></constructor-arg> <constructor-arg index="1"

  • 详解Java的MyBatis框架与Spring框架整合中的映射器注入

    MyBatis-Spring允许你在Service Bean中注入映射器.当使用映射器时,就像调用DAO那样来调用映射器就可以了,但是此时你就不需要进行任何DAO实现的编码,因为MyBatis会为你进行. 使用注入的映射器,你的代码就不会出现任何MyBatis-Spring依赖和MyBatis依赖.在我们的应用中有这样一个简单的映射器.你也应该知道映射器仅仅是一个接口: public interface UserMapper { User getUser(String userId); } 这是

随机推荐