170330、Spring中你不知道的注入方式
前言
在Spring配置文件中使用XML文件进行配置,实际上是让Spring执行了相应的代码,例如:
使用<bean>元素,实际上是让Spring执行无参或有参构造器
使用<property>元素,实际上是让Spring执行一次setter方法
但Java程序还可能有其他类型的语句:调用getter方法、调用普通方法、访问类或对象的Field等,而Spring也为这种语句提供了对应的配置语法:
调用getter方法:使用PropertyPathFactoryBean
调用类或对象的Filed值:使用FiledRetrievingFactoryBean
调用普通方法:使用MethodInvokingFactoryBean
注入其他Bean的属性值
PropertyPathFactoryBean用来获得目标Bean的属性值(实际上就是调用getter方法返回的值),获得的值可以注入给其他的Bean,也可以直接定义新的Bean。看如下的配置文件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
<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类的属性可以从配置文件中看出,这不再给出。主程序如下:
1
2
3
4
5
6
7
|
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()); } } |
输出结果:
1
|
age= 11 |
Bean实例的属性值,不仅可以注入另一个Bean,还可将Bean实例的属性值直接定义成Bean实例,这也是通过PropertyPathFactoryBean完成的。对上面的配置文件增加这样一段:
1
2
3
4
5
6
7
|
<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。下面是配置片段:
1
2
3
4
5
6
|
<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> |
测试主程序与上文定义的类似,这里不再提供,执行结果如下:
1
|
age= 8 |
在这个配置中,son对象的age的值,等于java.sql.Connection.TRANSACTION_SERIALIZABLE的值。在上面的定义中,定义FieldRetrievingFactoryBean工厂Bean时,指定的id并不是该Bean实例的唯一标识,而是指定Field的表达式(即将要被取出来的值)。
注意:Field既可以是静态的,也可以是非晶态的。上面的配置片段指定的Field表达式是静态Field值,因此可以通过类名直接访问。如果Field值是非静态的,则应该通过容器中已经存在的Bean来访问——即Field表达式的第一个短语应该是容器中已经存在的Bean。
Field值也可以定义成Bean实例,例如,在配置文件中增加下面一段:
1
2
3
4
5
6
7
|
<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> |
在主程序中增加如下输出:
1
|
System.out.println( "age=" + ac.getBean( "age" )); |
执行结果和上文一样。
使用FieldRetrievingFactoryBean获取Field值时,必须指定如下两个属性:
targetClass或targetObject:分别用于指定Field值所在的目标累或目标对象。如果需要获得的Field是静态的,则使用targetClass指定目标累;如果Field是非静态的,则使用targetObject指定目标对象
targetField:指定目标类或目标对象的Field名
如果Field是个静态Field,则有一种更加简洁的写法:
1
2
3
4
5
|
<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实例。看例子:
1
2
3
4
5
6
7
8
9
10
11
|
<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:
1
2
3
4
|
public class ValueGenerator { public int getValue() { return 2 ; } public static int getStaticValue () { return 3 ;} } |
测试程序依旧打印son1中age的值,代码略,结果如下:
1
|
age= 2 |
如果要调用静态方法,则把配置修改为:
1
2
3
4
5
6
7
8
9
10
|
<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> |
测试结果为:
1
|
age= 3 |
由于Java是支持重载的,只给定方法名,还不足以能够确定调用哪个方法,通过上面的配置能调用成功是因为ValueGenerator中的两个方法都没有参数。如果方法中有参数,该如何配置呢?在配置文件中加入以下内容:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
<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。即相当于:
1
|
javaVersion = java.lang.System.getProperty( "java.version" ); |
和前文中的Field一样,如果要调用的方法为静态方法,也有一种更加简洁的方法:
1
2
3
4
5
|
<bean id= "myBean" class = "org.springframework.beans.factory.config.MethodInvokingFactoryBean" > <!-- 使用staticMethod属性,直接指定目标类的目标方法 --> <property name= "staticMethod" value= "com.abc.util.ValueGenerator.getStaticValue" /> </bean>
|
170330、Spring中你不知道的注入方式的更多相关文章
- Java Spring 中你不知道的注入方式
前言 在Spring配置文件中使用XML文件进行配置,实际上是让Spring执行了相应的代码,例如: 使用<bean>元素,实际上是让Spring执行无参或有参构造器 使用<prop ...
- Spring中的注入方式 和使用的注解 详解
注解:http://www.cnblogs.com/liangxiaofeng/p/6390868.html 注入方式:http://www.cnblogs.com/java-class/p/4727 ...
- Spring中的注入方式
在Spring配置文件中使用XML文件进行配置,实际上是让Spring执行了相应的代码,例如: 使用<bean>元素,实际上是让Spring执行无参或有参构造器 使用<propert ...
- Spring中依赖注入的四种方式
在Spring容器中为一个bean配置依赖注入有三种方式: · 使用属性的setter方法注入 这是最常用的方式: · 使用构造器注入: · 使用Filed注入(用于注解方式). 使用属性的sett ...
- 使用IDEA详解Spring中依赖注入的类型(上)
使用IDEA详解Spring中依赖注入的类型(上) 在Spring中实现IoC容器的方法是依赖注入,依赖注入的作用是在使用Spring框架创建对象时动态地将其所依赖的对象(例如属性值)注入Bean组件 ...
- spring的set注入方式流程图解
spring的set注入方式流程图解 自己学习spring的一些笔记,详细画出了spring的set方式实现依赖注入的流程. 注意:<property name="UserDao&qu ...
- spring中构造函数注入
spring中构造函数注入,简单来说,就是通过beans.xml中,设置对应的值.而且通过bean类中的构造函数进行注入这些值. 文件结构 watermark/2/text/aHR0cDovL2Jsb ...
- Spring中属性注入的几种方式以及复杂属性的注入
在Spring框架中,属性的注入我们有多种方式,我们可以通过构造方法注入,可以通过set方法注入,也可以通过p名称空间注入,方式多种多样,对于复杂的数据类型比如对象.数组.List集合.map集合.P ...
- Spring中属性注入的几种方式以及复杂属性的注入详解
在spring框架中,属性的注入我们有多种方式,我们可以通过set方法注入,可以通过构造方法注入,也可以通过p名称空间注入,方式多种多样,对于复杂的数据类型比如对象.数组.List.Map.Prope ...
随机推荐
- 以源码编译的方式安装PHP与php-fpm
首先是最基本的下载,解压,编译安装(以PHP 5.3.6 为例): wget http://www.php.net/get/php-5.3.6.tar.gz/from/this/mirrortar x ...
- Redis(十二):redis两种持久化方法对比分析
前言 最近在项目中使用到Redis做缓存,方便多个业务进程之间共享数据.由于Redis的数据都存放在内存中,如果没有配置持久化,redis重启后数据就全丢失了,于是需要开启redis的持久化功能,将数 ...
- C数组逆序
一.标准交换模式 /**** *标准交换模式 *实现数组的逆序,原理就是数组的首尾元素进行交换 ***/ #define N 5; int main(){ int array[N] = {15,20, ...
- C++和C#实现剪切板数据交互
c#端由于system.windows.form自带的剪切板功能太少,所以写了一个Helper类把接口转了出来.这样就可以用不同的uint的id了. 并且自带的剪切板必须执行在[STAThread]模 ...
- atitit.商业版 源码保护 与 java本地原生代码转换 的方案总结
atitit.商业版 源码保护 与 java本地原生代码转换 的方案总结 1. 为什么虚拟机语言容易被反编译 1 2. 源码泄露的问题问题 1 3. Excelsior JET 1 4. gcj.的流 ...
- 每日英语:China Grapples With Genetically Modified Foods
A Chinese agricultural official's unsupported claims about the carcinogenic risks of consuming genet ...
- 总结iOS9中的新的方法
iOS平台在快速的发展,各种接口正在不断的更新.随着iOS9的发布,又有一批老方法不推荐使用了,你若调用这些方法,运行的结果是没有问题的,但是会出现警告“***is deprecated :first ...
- IIS网站本机可以访问,局域网其他机器无法访问
[开始]打开[控制面板],选择[WINDOWS 防火墙],进入 步骤阅读 2 在防火墙界面,选择左侧边栏的[高级设置] 步骤阅读 3 在弹出的高级安全Windows防火墙界面中,选择左侧边栏的[入站规 ...
- word字号
1 大特号 63 2 特 号 54 3 初 号 42 4 小初号 36 5 大一号 31.5 6 一 号 28 7 小一号 24 8 二 号 21 9 小二号 18 10 三 号 16 11 小三号 ...
- js监听文本框变化事件
用js有两种写法: 法一: <!DOCTYPE HTMl> <html> <head> <title> new document </title& ...