@Autowired用法详解

本文详细介绍了@Autowired注解在Spring框架中的应用,涵盖了自动装配的原理、如何简化bean配置、@Autowired与构造方法的执行顺序,以及如何处理多实例匹配。通过实例演示,展示了如何利用Qualifier注解进行精确匹配。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

@Autowired 注解简介

@Autowired 注解,它可以对类成员变量、方法及构造函数进行标注,完成自动装配的工作。 通过 @Autowired的使用来消除 set ,get方法。在使用@Autowired之前,我们对一个bean配置起属性时,是这样用的

<property name="属性名" value=" 属性值"/>   

通过这种方式来,配置比较繁琐,而且代码比较多。在Spring 2.5 引入了 @Autowired 注解。
下面用案例来具体说明
UserRepository.java

package com.proc.bean.repository;

public interface UserRepository {
     
     void save();
}

这里定义了一个UserRepository接口,其中定义了一个save方法

UserRepositoryImps.java

package com.proc.bean.repository;

import org.springframework.stereotype.Repository;
 
@Repository("userRepository")//指定了该bean在IoC中标识符名称为userRepository
public class UserRepositoryImps implements UserRepository{
    @Override
    public void save() {
        System.out.println("UserRepositoryImps save");
    }
}

定义一个UserRepository接口的实现类,并实现save方法,注意!在这里指定了该bean在IoC中标识符名称为userRepository

UserService.java

package com.proc.bean.service;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
 
import com.proc.bean.repository.UserRepository;
 
@Service
public class UserService {
 
    @Autowired
    private UserRepository userRepository;
    public void save(){
        userRepository.save();
    }
}

这里需要一个UserRepository类型的属性实例,通过@Autowired自动装配方式,从IoC容器中去查找到,并返回实例

applicationContext.xml配置

<context:component-scan base-package="com.proc.bean" />

测试代码:

ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService=(UserService) ctx.getBean("userService");
userService.save(); //结果为UserRepositoryImps save

@Autowired的原理

其实在启动spring IoC时,容器自动装载了一个AutowiredAnnotationBeanPostProcessor后置处理器,当容器扫描到@Autowied、@Resource或@Inject时,就会在IoC容器自动查找需要的bean,并装配给该对象的属性

 <bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor"/>

注意事项:
在使用@Autowired时,首先在容器中查询对应类型的bean。如果查询结果刚好为一个,就将该bean装配给@Autowired指定的数据如果查询的结果不止一个,那么@Autowired会根据名称来查找。如果查询的结果为空,那么会抛出异常。解决方法时,使用required=false
举例说明: 接着上面的例子,我们在定一个类来实现UserRepository接口

package com.proc.bean.repository;

import org.springframework.stereotype.Repository;

@Repository
public class UserJdbcImps implements UserRepository {
    @Override
    public void save() {
        System.out.println("UserJdbcImps save");
    }
}

这时在启动容器后,在容器中有两个UserRepository类型的实例,一个名称为userRepository,另一个为userJdbcImps。在UserService中

@Autowired
private UserRepository userRepository; // 输出结果:UserRepositoryImps save

这里由于查询到有两个该类型的实例,那么采用名称匹配方式,在容器中查找名称为userRepository的实例,并自动装配给该参数。如果这里想要装配userJdbcImps的实例,除了可以将字段userRepository名称改成userJdbcImps外,还可以提供了一个@Qualifier注解,来指定需要装配bean的名称,代码如下:

@Autowired
@Qualifier("userJdbcImps")
private UserRepository userRepository; //输出结果:UserJdbcImps save

上文链接

注意:每个类可以有很多个构造方法,但是在使用@Autowired的时候,有且只能有一个构造方法能够被标记为required=true(注:required的默认值为true)。

@Autowired和构造方法执行顺序差异

下面这段代码会报错:

@Autowired
private Person person;
private Dog dog;

public ServiceImpl(){
	this.dog= person.dog();
}

之所以报这个错是因为Java类会先执行构造方法,然后再给注解了@Autowired 的user注入值。Java变量的初始化顺序:
静态变量或静态语句块–>实例变量或初始化语句块–>构造方法–>@Autowired
所以,想要给user值注入,我们可以通过构造方法注入


private Person person;
private String dog;

@Autowired
public ServiceImpl(Person person){
	this.person= person;
	this.dog= person.getdog();
}

使用构造方法注入,可以明确成员变量的加载顺序。

### Spring 中 `@Autowired` 注解的使用说明 #### 1. 自动装配的基础机制 在 Spring 框架中,`@Autowired` 是一种基于依赖注入(DI)的核心注解。它允许开发者声明组件之间的关系,并由 Spring 容器负责管理这些依赖项的实例化和绑定过程[^1]。 #### 2. 构造函数注入 从 Spring 4.3 开始,如果某个 Bean 只有一个构造函数,则无需显式标记 `@Autowired` 注解,Spring 将自动识别并完成依赖注入操作。然而,在存在多个构造函数的情况下,必须指定其中一个构造函数并通过 `@Autowired` 进行标注以消除歧义[^2]。 ```java public class MyService { private final MyRepository myRepository; // 单一构造函数情况下不需要加 @Autowired public MyService(MyRepository myRepository) { this.myRepository = myRepository; } } ``` #### 3. Setter 方法注入 除了通过构造函数实现依赖注入外,还可以利用 setter 方法来达到相同目的。在这种方式下,只需简单地将 `@Autowired` 添加至对应的 setter 方法即可。 ```java private MyRepository myRepository; @Autowired public void setMyRepository(MyRepository myRepository) { this.myRepository = myRepository; } ``` #### 4. 字段直接注入 字段级别的注入是最常见也是最简洁的方式之一。只需要把 `@Autowired` 放置于需要被注入的对象变量前就可以让 Spring 来处理具体的赋值逻辑[^4]。 ```java @Autowired private MyRepository myRepository; ``` #### 5. 解决多候选 bean 的冲突问题 当项目中有多个同类型的 bean 存在于 IOC 容器里时,可能会引发无法决定具体应该选用哪个 bean 的情况。此时可以通过引入 `@Qualifier` 注解进一步限定特定名称的 bean 实例[^3]。 ```java @Autowired @Qualifier("specificBeanName") private MyInterface specificBean; ``` 或者采用更现代的做法——定义 Primary 或者使用 JSR-330 标准下的 `@Named` 和 `@Inject` 组合替代传统写法。 --- ### 常见问题解答 #### Q: 如果有多个符合条件的 Beans 怎么办? A: 当遇到这种情形时,可以借助于 `@Primary`, `@Qualifier` 等额外约束条件加以区分;另外也可以考虑重新设计应用程序结构减少不必要的重复定义。 #### Q: 是否可以在接口上使用 @Autowired? A: 不可以直接作用于抽象方法所在的接口之上,因为实际上并没有提供任何可供反射解析的具体实现细节给 IoC 容器参考。因此通常只会在 concrete classes 上面运用此特性。 #### Q: 如何验证所有的 @Autowired 都成功完成了初始化? A: 设置属性 required=false 虽然可以让程序继续运行下去即使某些地方缺少必要的配置文件支持,但从长远来看并不推荐这样做。更好的办法是在测试阶段充分覆盖各种边界场景确保整个系统的健壮性。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值