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 >
目录
相关文章
|
4月前
|
XML Java 程序员
Spring6框架中依赖注入的多种方式(推荐构造器注入)
依赖注入(DI)是一种过程,对象通过构造函数参数、工厂方法的参数或在对象实例构建后设置的属性来定义它们的依赖关系(即与其一起工作的其他对象)。
64 3
|
1月前
|
XML Java 数据格式
Spring5入门到实战------4、IOC容器-Bean管理XML方式、集合的注入(二)
这篇文章是Spring5框架的实战教程,主题是IOC容器中Bean的集合属性注入,通过XML配置方式。文章详细讲解了如何在Spring中注入数组、List、Map和Set类型的集合属性,并提供了相应的XML配置示例和Java类定义。此外,还介绍了如何在集合中注入对象类型值,以及如何使用Spring的util命名空间来实现集合的复用。最后,通过测试代码和结果展示了注入效果。
Spring5入门到实战------4、IOC容器-Bean管理XML方式、集合的注入(二)
|
22天前
|
缓存 Java 数据库连接
Spring Boot 资源文件属性配置,紧跟技术热点,为你的应用注入灵动活力!
【8月更文挑战第29天】在Spring Boot开发中,资源文件属性配置至关重要,它让开发者能灵活定制应用行为而不改动代码,极大提升了可维护性和扩展性。Spring Boot支持多种配置文件类型,如`application.properties`和`application.yml`,分别位于项目的resources目录下。`.properties`文件采用键值对形式,而`yml`文件则具有更清晰的层次结构,适合复杂配置。此外,Spring Boot还支持占位符引用和其他外部来源的属性值,便于不同环境下覆盖默认配置。通过合理配置,应用能快速适应各种环境与需求变化。
27 0
|
23天前
|
安全 Java 开发者
开发者必看!@Resource与private final的较量,Spring Boot注入技巧大揭秘,你不可不知的细节!
【8月更文挑战第29天】Spring Boot作为热门Java框架,其依赖注入机制备受关注。本文通过对比@Resource(JSR-250规范)和@Autowired(Spring特有),并结合private final声明的字段注入,详细探讨了两者的区别与应用场景。通过示例代码展示了@Resource按名称注入及@Autowired按类型注入的特点,并分析了它们在注入时机、依赖性、线程安全性和单一职责原则方面的差异,帮助开发者根据具体需求选择最合适的注入策略。
30 0
|
2月前
|
Java Spring
spring注入的几种方式
spring注入的几种方式
18 0
|
3月前
|
Java Spring 容器
spring如何进行依赖注入,通过set方法把Dao注入到serves
spring如何进行依赖注入,通过set方法把Dao注入到serves
|
3月前
spring-boot报错循环注入报错:has been injected into other beans
spring-boot报错循环注入报错:has been injected into other beans
206 3
|
3月前
|
Java Linux 程序员
技术笔记:Spring生态研习【五】:Springboot中bean的条件注入
技术笔记:Spring生态研习【五】:Springboot中bean的条件注入
|
4月前
|
前端开发 Java 编译器
详解Spring与JDK注入
依赖注入是Spring框架的核心概念之一,它通过将对象之间的依赖关系外部化,实现了松耦合和可测试性。面向切面编程则允许开发人员将横切关注点(如日志、事务管理)从应用程序的主要业务逻辑中分离出来,以提高代码的模块化和可维护性。
44 4
|
3月前
|
SQL 安全 Java
Spring Boot中的跨站点脚本攻击(XSS)与SQL注入防护
【6月更文挑战第15天】在现代Web应用程序开发中,安全性是一个至关重要的课题。跨站点脚本攻击(XSS)和SQL注入是最常见的两种攻击类型,它们可以严重威胁到应用程序的安全。
456 0