@AutoWired与@Resource(以及@Qualifier)

spring不但支持自己定义的@Autowired注解(所以Autowired与Spring是强相关性,只能在spring框架中使用,而后几个注解则不然),还支持几个由JSR-250规范定义的注解,它们分别是@Resource、@PostConstruct以及@PreDestroy。

@PostConstruct

相当于init-method,使用在方法上,当Bean初始化时执行,执行顺序在@Autowire注解之后。

@PreDestroy

相当于destory-method,使用在方法上,当Bean销毁时执行。

@Autowired

  1. @Autowired 注解可以用于 Setter 方法字段、构造函数。(也可用于普通方法,前提是方法必须有至少一个参数,很少用,此时会产生一个副作用,就是在容器初始化该 Bean 实例的时候就会调用该方法。当然,前提是执行了自动装配,对于不满足装配条件的情况,该方法也不会被执行)。

附:最近见到了不少将此注解标注在构造函数上的情况,如以下场景:
类A需要在初始化立刻执行某个操作method,但此操作需要必须在属性attribute初始化之后才能进行,方案有多种
1.实现InitializingBean接口,在afterPropertiesSet方法中执行method
2.实现ApplicationListener<ContextRefreshedEvent>接口,在onApplicationEvent方法中执行method
3.使用@postConstruct注解method方法,此注解标注的方法会在所有属性注入完成后,获取此bean前执行
4.属性attribute不使用自动装配,使用@Autowired注解构造函数,并注入attribute

@Autowired
 public A(Attribute attribute) {
	 this.attribute = attribute;
	 method();
 } 
  1. @Autowired按byType自动注入,查找指定类型的Bean。找不到则抛出异常(可以给 @Autowired 标注增加一个 required=false 属性,以改变这个行为)。 如果找到多个bean,按以下方案进行:
  1. 选择其中带有Primary注解的bean,如果只有一个直接注入,如果有多个bean带有Primary注解则报错。
  2. 选择与@Qualifier注解的value值名称相同的Bean,如下,如果有名为zhangsan的User Bean,则注入,无则报错
	@Autowired 
 	@Qualifier("zhangsan")
 	public User user;

    // 注解在set方法上的例子
    @Autowired   
    public void setUserDao(@Qualifier("userDao") UserDao userDao) {   
        this.userDao = userDao;   
    }  

@Resource

  1. @Resource默认按 byName自动注入。@Resource有两个属性是比较重要的,分是name和type,Spring将@Resource注解的name属性解析为bean的名字,而type属性则解析为bean的类型。所以如果使用name属性,则使用byName的自动注入策略(同时也会判断该name的Bean是否类型匹配,不存在或不匹配将抛出异常),而使用type属性时则使用byType自动注入策略。如果既不指定name也不指定type属性,这时将通过反射机制使用byName自动注入策略。
  2. Resource装配顺序:
    2.1 如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常
    2.2 如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常
    2.3 如果指定了type,则从上下文中找到类型匹配的唯一bean进行装配,找不到或者找到多个,都会抛出异常
    2.4 如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果成功则自动装配;

其他扩展

  1. 常见定义Bean的注解:

@Controller

@Controller(“Bean的名称”)

定义控制层Bean,如Action

@Service

@Service(“Bean的名称”)

定义业务层Bean

@Repository

@Repository(“Bean的名称”)

定义DAO层Bean

@Component

它是一个通用泛化的注解,当某个组件不能通过业务进行归类时,可以使用这个注解进行标注。@Controller、@Repository、@Service是基于此注解的扩展,查看源码可以发现这三个注解上都标注了@Component 注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Controller {
    @AliasFor(
        annotation = Component.class
    )
    String value() default "";
}

以上几个注解效果是完全一样的,只是后三个用于方便标示该Bean的业务范围。

注:这几个注解都依赖于类的无参构造,如果未提供,则会在启动时报错

No default constructor found; nested exception is java.lang.NoSuchMethodException

### Spring 中 `@Autowired` `@Resource` 的区别及其使用场景 #### 主要区别 1. **来源不同** - `@Autowired` 是由 Spring 提供的注解,主要用于 Spring 容器中的依赖注入[^1]。 - `@Resource` 则是由 JDK 提供的标准注解,遵循 JSR-250 规范,适用于任何支持该规范的 Java 容器[^4]。 2. **默认注入方式** - `@Autowired` 默认按照类型 (`ByType`) 进行匹配并完成 Bean 的注入。如果存在多个相同类型的 Bean,则会抛出异常。 - `@Resource` 默认按照名称 (`ByName`) 进行匹配。如果没有找到对应的名称,则退而求其次按类型进行匹配。 3. **解决多实现类的情况** - 当一个接口有多个实现类时: - 对于 `@Autowired`,可以配合 `@Qualifier` 明确指定需要注入的具体 Bean 名称。 - 而对于 `@Resource`,可以直接通过其属性 `name` 来显式指定目标 Bean 的名称。 4. **灵活性控制力** - 在 Spring 应用开发中,推荐优先使用 `@Autowired`,因为它更加贴合 Spring 的设计理念,并提供诸如懒加载(`lazy-init=true`)以及更精细的错误处理机制等功能[^2]。 #### 使用场景分析 - 如果项目完全基于 Spring 或者主要采用 Spring 技术栈构建,则应首选 `@Autowired` 实现依赖注入操作。 - 若处于混合技术环境或者希望保持代码独立性(不绑定特定框架),那么可以选择更为通用的 `@Resource` 注解。 ```java // 示例:使用 @Autowired 配合 @Qualifier 处理多实现情况 @Component public class ExampleService { @Autowired @Qualifier("specificImplementation") private MyInterface myInstance; } // 示例:使用 @Resource 指定具体 bean name @Component public class AnotherExampleService { @Resource(name="anotherSpecificImplementation") private MyInterface anotherMyInstance; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值