[Spring实战系列](13)使用注解自动装配

简介: 版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SunnyYoona/article/details/50644820 1. 简介从Spring2.5开始,我们就可以使用注解的自动装配方式装配Spring Bean的属性。
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SunnyYoona/article/details/50644820

1. 简介

从Spring2.5开始,我们就可以使用注解的自动装配方式装配Spring Bean的属性。使用注解自动装配方式与在XML中使用autowire属性自动装配没有太大区别。那为啥还要研发出这样一种装配方式?肯定有它独特的地方: 使用注解自动装配方式允许更细粒度的自动装配,我们可以选择性的标注某一个属性对其应用自动装配

2. 启用注解装配


Spring容器 默认禁用注解装配。所以,在使用基于注解的自动装配前,我们需要 在Spring配置中启用它。最简单的启用方式是 使用Spring的context命名空间配置的<context:annotation-config>元素

 
   
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
 
</beans>

<context:annotation-config/>元素告诉Spring我们打算使用基于注解的自动装配。一旦装配完成,我们就可以对代码添加注解,表示Spring应该为属性,方法和构造器进行自动装配。

3. 使用@Autowired


 
   
@Autowired
public void setSchool(School school) {
this.school = school;
}

当我们对setSchool()方法使用 @Autowired注解 时,我们可以移除用来定义Student的school属性所对应的<property>元素了。如下面配置文件中的yoona Bean。

 
   
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
 
<bean id = "yoona" class = "com.sjf.bean.Student">
<property name="name" value="yoona"/>
<property name="age" value="24"/>
</bean>
 
<bean id = "xidianSchool" class = "com.sjf.bean.School">
<property name="name" value="西安电子科技大学"/>
<property name="location" value="西安"/>
</bean>
 
</beans>

运行结果:

name:yoona   age:24   school:西安电子科技大学[西安]

在这个实例中,我们对Student的school属性使用 使用 @Autowired注解 ,name属性和age属性使用setter注入方式。 当Spring发现我们对setSchool()方法使用 @Autowired注解 时,Spring就会尝试对该方法 执行byType自动装配 。查询到只有一个Bean(xidianSchool)满足其条件,自动把xidianSchool Bean注入到Student的school属性中。

注意点:

使用了@Autowired注解时,Spring就会尝试对该方法执行byType自动装配

特别之处(不同XML方式):

使用@Autowired注解不只可以使用它标注setter方法,还可以标注需要自动装配Bean引用的任意方法。甚至是构造器,@Autowired注解表示当创建Bean时,即使Spring XML文件中没有使用<constructor-arg>元素配置Bean,该构造器也需要进行自动装配。Spring会从所有满足装配条件的构造器中选择入参最多的那个构造器。还可以标注属性,并可以删除setter方法

(1) 使用@Autowired注解标注setter方法:
 
   
@Autowired
public void setSchool(School school) {
this.school = school;
}
(2)使用@Autowired注解标注属性(不受限于private关键字):
 
   
@Autowired
private School school;
(3)使用@Autowired注解标注任意方法:
 
   
@Autowired
public void assemblySchool(School school) {
this.school = school;
}
(4)使用@Autowired注解标注构造器:
 
   
@Autowired
public Student(School school) {
this.school = school;
}

局限性(同于XML方式):

应用中必须只能有一个Bean适合装配到@Autowired注解所标注的属性或者参数中。如果没有匹配的Bean,或者存在多个匹配的Bean,@Autowired注解就会遇到一些麻烦。
3.1 没有匹配的Bean

如果没有匹配的Bean,@Autowired注解就会失败,抛出 NoSuchBeanDefinitionException 异常

Caused by:  org.springframework.beans.factory.NoSuchBeanDefinitionException
No qualifying bean of type [com.sjf.bean.School] found for dependency: expected 
at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {}
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.
raiseNoSuchBeanDefinitionException( DefaultListableBeanFactory.java:1373 )
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.
doResolveDependency( DefaultListableBeanFactory.java:1119 )
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.
resolveDependency( DefaultListableBeanFactory.java:1014 )
    at org.springframework.beans.factory.annotation.
AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.inject( AutowiredAnnotationBeanPostProcessor.java:618 )
    ... 15 more  
属性不一定非要装配,null值也是可以接受的。这种情况下,我们可以设置@Autowired的required 属性为false来装配自动装配的是可选的。
 
   
@Autowired(required = false)
public void setSchool(School school) {
this.school = school;
}

在这里,Spring将尝试装配school属性,但是如果没有查找到与之匹配的类型为School的Bean,不会抛出任何异常,并且Student Bean的school属性会设置为null。

注意点:

required属性可以用于@Autowired注解所使用的任意地方。但是当使用构造器装配时,只有一个构造器可以将@Autowired的required属性设置true其他使用@Autowired注解所标注的构造器只能将required属性设置为false
3.2 多个匹配的Bean

另一个问题是可能会有足够多的(至少2个)都满足装配条件,并且都可以被装配到属性或参数中。假如,我们有两个Bean是School类型的,满足装配条件。这种情况下,@Autowired注解没有办法选择哪一个Bean才是它需要的,会抛出 NoUniqueBeanDefinitionException 异常。

 
   
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
 
<bean id = "yoona" class = "com.sjf.bean.Student">
<property name="name" value="yoona"/>
<property name="age" value="24"/>
</bean>
 
<bean id = "xidianSchool" class = "com.sjf.bean.School">
<property name="name" value="西安电子科技大学"/>
<property name="location" value="西安"/>
</bean>
 
<bean id = "shandaSchool" class = "com.sjf.bean.School">
<property name="name" value="山东大学"/>
<property name="location" value="山东"/>
</bean>
</beans>
抛出异常:
Caused by:  org.springframework.beans.factory.NoUniqueBeanDefinitionException
No qualifying bean of type [com.sjf.bean.School] is defined: expected single matching bean but found 2: xidianSchool,shandaSchool
  at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency( DefaultListableBeanFactory.java:1126 )
  at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency( DefaultListableBeanFactory.java:1014 )
  at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.
inject( AutowiredAnnotationBeanPostProcessor.java:618 )
  ... 15 more

为了帮助@Autowired鉴别出哪一个Bean才是我们所需要的,我们可以配合使用Spring的 @Qualifier注解。
 
   
@Autowired
@Qualifier("shandaSchool")
public void setSchool(School school) {
this.school = school;
}
这样,我们就使用@Qualifier注解尝试注入ID为"shandaSchool"的Bean。表面上看,把@Autowired的byType自动装配转换为显示的byName装配。实际只是使用 @Qualifier注解缩小了自动装配挑选候选Bean的范围。上例中通过指定Bean 的ID把选择范围缩小到只剩一个Bean。

除了通过Bean的ID缩小选择范围,还可以通过在Bean上直接使用qualifier来缩小范围。
 
   
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<context:annotation-config/>
 
<bean id = "yoona" class = "com.sjf.bean.Student">
<property name="name" value="yoona"/>
<property name="age" value="24"/>
</bean>
 
<bean id = "xidianSchool" class = "com.sjf.bean.School">
<property name="name" value="西安电子科技大学"/>
<property name="location" value="西安"/>
<qualifier value="ShanXi"/>
</bean>
 
<bean id = "shandaSchool" class = "com.sjf.bean.School">
<property name="name" value="山东大学"/>
<property name="location" value="山东"/>
<qualifier value="ShanDong"/>
</bean>
</beans>
这里,限定shandongSchool Bean是山东地区的大学,xidianSchool Bean是陕西地区的大学。
 
   
@Autowired
@Qualifier("ShanXi")
public void setSchool(School school) {
this.school = school;
}

运行结果:

name:yoona   age:24   school:西安电子科技大学[西安]

这里通过指定@Qualifier属性为"ShanXi",把选择范围缩小到只剩一个Bean(xidianSchool)。


4. 在注解注入中使用表达式

既然可以使用注解为Spring的Bean自动装配其他Bean的引用,我们同样希望能够 使用注解装配简单的值。Spring3.0 引入了 @Value,它是一个新的装配注解。可以让我们使用注解装配String类型的值和基本类型的值,例如int,boolean。可以 通过@Value直接标注某个属性,方法或者方法参数,并传入一个String类型的表达式来装配属性
 
   
@Value("yoona")
private String name;
在这里,我们为String类型的属性装配了一个String类型的值。但是传入@Value的String类型的参数只是一个表达式,它的计算结果可以是任意类型,因此@Value能够标注任意类型的属性。

实际上,装配简单值,不是@Value的主要作用,它可以借助SEL表达式,作用更加突出。我们知道运行期可以通过 SpEL动态计算复杂表达式的值并把结果装配到Bean的属性中。这一特性也使得 @Value注解成为强大的装配可选方案。

 
   
@Value("#{systemProperties.myFavoriteSong}")
private String song;
在这个例子中,我们使用SpEL从系统属性中获取一个值。



参考:《Spring实战》



目录
相关文章
|
17天前
|
Java Spring
在使用Spring的`@Value`注解注入属性值时,有一些特殊字符需要注意
【10月更文挑战第9天】在使用Spring的`@Value`注解注入属性值时,需注意一些特殊字符的正确处理方法,包括空格、引号、反斜杠、新行、制表符、逗号、大括号、$、百分号及其他特殊字符。通过适当包裹或转义,确保这些字符能被正确解析和注入。
|
5天前
|
XML JSON Java
SpringBoot必须掌握的常用注解!
SpringBoot必须掌握的常用注解!
23 4
SpringBoot必须掌握的常用注解!
|
7天前
|
存储 缓存 Java
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
41 2
|
7天前
|
JSON Java 数据库
SpringBoot项目使用AOP及自定义注解保存操作日志
SpringBoot项目使用AOP及自定义注解保存操作日志
25 1
|
15天前
|
自然语言处理 Java API
Spring Boot 接入大模型实战:通义千问赋能智能应用快速构建
【10月更文挑战第23天】在人工智能(AI)技术飞速发展的今天,大模型如通义千问(阿里云推出的生成式对话引擎)等已成为推动智能应用创新的重要力量。然而,对于许多开发者而言,如何高效、便捷地接入这些大模型并构建出功能丰富的智能应用仍是一个挑战。
63 6
|
19天前
|
缓存 NoSQL Java
Spring Boot与Redis:整合与实战
【10月更文挑战第15天】本文介绍了如何在Spring Boot项目中整合Redis,通过一个电商商品推荐系统的案例,详细展示了从添加依赖、配置连接信息到创建配置类的具体步骤。实战部分演示了如何利用Redis缓存提高系统响应速度,减少数据库访问压力,从而提升用户体验。
50 2
|
22天前
|
架构师 Java 开发者
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
在40岁老架构师尼恩的读者交流群中,近期多位读者成功获得了知名互联网企业的面试机会,如得物、阿里、滴滴等。然而,面对“Spring Boot自动装配机制”等核心面试题,部分读者因准备不足而未能顺利通过。为此,尼恩团队将系统化梳理和总结这一主题,帮助大家全面提升技术水平,让面试官“爱到不能自已”。
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
|
2天前
|
存储 安全 Java
springboot当中ConfigurationProperties注解作用跟数据库存入有啥区别
`@ConfigurationProperties`注解和数据库存储配置信息各有优劣,适用于不同的应用场景。`@ConfigurationProperties`提供了类型安全和模块化的配置管理方式,适合静态和简单配置。而数据库存储配置信息提供了动态更新和集中管理的能力,适合需要频繁变化和集中管理的配置需求。在实际项目中,可以根据具体需求选择合适的配置管理方式,或者结合使用这两种方式,实现灵活高效的配置管理。
6 0
|
26天前
|
XML Java 数据库
Spring boot的最全注解
Spring boot的最全注解
|
14天前
|
存储 Java 数据管理
强大!用 @Audited 注解增强 Spring Boot 应用,打造健壮的数据审计功能
本文深入介绍了如何在Spring Boot应用中使用`@Audited`注解和`spring-data-envers`实现数据审计功能,涵盖从添加依赖、配置实体类到查询审计数据的具体步骤,助力开发人员构建更加透明、合规的应用系统。