Spring源码学习之:你不知道的spring注入方式

简介: 前言    在Spring配置文件中使用XML文件进行配置,实际上是让Spring执行了相应的代码,例如:使用元素,实际上是让Spring执行无参或有参构造器使用元素,实际上是让Spring执行一次setter方法    但Java程序还可能有其他类型的语句:调用getter方...

前言

    在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>
相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
68 2
|
1月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
2月前
|
Java Spring
在使用Spring的`@Value`注解注入属性值时,有一些特殊字符需要注意
【10月更文挑战第9天】在使用Spring的`@Value`注解注入属性值时,需注意一些特殊字符的正确处理方法,包括空格、引号、反斜杠、新行、制表符、逗号、大括号、$、百分号及其他特殊字符。通过适当包裹或转义,确保这些字符能被正确解析和注入。
121 3
|
8天前
|
Java Spring
一键注入 Spring 成员变量,顺序编程
介绍了一款针对Spring框架开发的插件,旨在解决开发中频繁滚动查找成员变量注入位置的问题。通过一键操作(如Ctrl+1),该插件可自动在类顶部添加`@Autowired`注解及其成员变量声明,同时保持光标位置不变,有效提升开发效率和代码编写流畅度。适用于IntelliJ IDEA 2023及以上版本。
一键注入 Spring 成员变量,顺序编程
|
14天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
30 2
|
1月前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
62 9
|
2月前
|
前端开发 Java 数据库
SpringBoot学习
【10月更文挑战第7天】Spring学习
39 9
|
26天前
|
Java Kotlin 索引
学习Spring框架特性及jiar包下载
Spring 5作为最新版本,更新了JDK基线至8,修订了核心框架,增强了反射和接口功能,支持响应式编程及Kotlin语言,引入了函数式Web框架,并提升了测试功能。Spring框架可在其官网下载,包括文档、jar包和XML Schema文档,适用于Java SE和Java EE项目。
29 0
|
2月前
|
XML Java 数据格式
Spring学习
【10月更文挑战第6天】Spring学习
23 1
|
2月前
|
Java 测试技术 开发者
springboot学习四:Spring Boot profile多环境配置、devtools热部署
这篇文章主要介绍了如何在Spring Boot中进行多环境配置以及如何整合DevTools实现热部署,以提高开发效率。
97 2