@Autowired注解的实现原理和使用方法

发布于 2021-12-08  65 次阅读


  • 1.Spring框架中的@Autowired注解可以标注在哪些地方?

@Autowired注解可以被标注在构造函数、属性、setter方法或配置方法上,用于实现依赖自动注入。

  • 2.@Autowired注解的工作原理?

点开注解,可以看到相关的英文注释如图:

@Autowired注解的作用是由AutowiredAnnotationBeanPostProcessor实现的,查看该类的源码会发现它实现了MergedBeanDefinitionPostProcessor接口,进而实现了接口中的postProcessMergedBeanDefinition方法,@Autowired注解正是通过这个方法实现注入类型的预解析,将需要依赖注入的属性信息封装到InjectionMetadata类中,InjectionMetadata类中包含了哪些需要注入的元素及元素要注入到哪个目标类中,在Spring容器启动的过程中初始化单例bean的时候通过populateBean方法实现对属性的注入。

Spring容器在启动的时候会执行AbstractApplicationContext类的refresh方法,在refresh方法执行的过程中先注册AutowiredAnnotationBeanPostProcessor,然后在对非延迟初始化的单例bean进行初始化时,会间接调用。

  • 4.注入的过程?

使用AutowiredFieldElement实现对标注在属性上的注入,使用AutowiredMethodElement对标注在方法上的注入。注入过程:根据需要注入的元素的描述信息,按类型或名称查找需要的依赖值,如果依赖没有实例化先实例化依赖,然后使用反射进行赋值。

  • 5.@Resource或者@Autowired注解有什么区别?

虽然@Resource和@Autowired都可以书写标注在属性或者该属性的setter方法之上,但是@Resource默认是按照名称来装配注入的,只有当找不到与名称匹配的bean才会按照类型来装配注入;@Autowired默认是按照类型装配注入的,默认情况下它要求依赖对象必须存在如果允许为null,可以设置它required属性为false,如果想按照名称来注入,则需要结合@Qualifier一起使用;@Resource注解是由JDK提供,而@Autowired是由Spring提供。

具体使用方法:

@Autowired 字段

我们可以把 @Autowired 注解标注在类文件中的字段属性上,通过这种方式,Spring 容器启动的时候会查找相应的 Bean ,然后通过反射的方式注入到这个字段中。

例子如下:

@Controller
public class PayController {
    @Autowired
    PayService aliPayService;
}

这种方式使用起来非常方便,而且对于字段的要求也比较低,它可以是 public,也可以是 private范围。

日常编程中,我一般是直接使用这种方式,简洁又便利。

@Autowired 构造方法

第二种方式,我们可以使用类中的构造函数注入相应属性,例子如下:

@Controller
public class PayController {
    
    PayService aliPayService;
    
    @Autowired
    public PayController(PayService aliPayService) {
        this.aliPayService = aliPayService;
    }
}

@Autowired 构造方法里的传参,可以是一个,也可以是多个。如果是多个参数,那 Spring 一起都注入进来。

@Controller
public class PayController {

    PayService aliPayService;
    
    OrderService orderService;
    
    @Autowired
    public PayController(PayService aliPayService, OrderService orderService) {
        this.aliPayService = aliPayService;
        this.orderService=orderService;
    }
}

这种方式在注入多个属性的时候用起来还是比较方便的。

那其实有一种情况下,如果一个类不存在无参的构造函数,只存在有参的构造函数,那实际上我们即使不用 @Autowired标注,Spring 也会帮我们注入相应的属性。

可以看到,上面的例子,我们没有在构造函数上使用 @Autowired标注,但是 IDEA 依然显示我们成功注入相应的属性。

@Autowired 方法

第三种方式,我们可以通过方法注入相应属性,例子如下:

@Controller
public class PayController {

    PayService aliPayService;
    
    @Autowired
    public void setAliPayService(PayService aliPayService) {
        this.aliPayService = aliPayService;
    }
}

那之前我一直以为这种方式只能使用  setxxx才可以成功的注入属性 ,那其实是我理解错了,这个方法可以任意的命名。

那方法注入,其实跟构造方法一样,也可以同时有多个传参,Spring 将会把这些属性一起注入。

@Autowired  多个 Bean

最后一种方式,Spring 可以使用 @Autowired 标注 Array (数组),Collections(集合),甚至是 Map (散列表),通过这种命方式注入多个相同类型的 Bean

比如这种方式,将 PayService 接口所有实现类注入到 Array数组中。

也可以是这种方式,注入到 List 集合中。

上面的两个例子,数组与集合中的 Bean 的顺序是根据 Spring 创建的顺序。

如果你想指定里面排序的优先级,那你就需要使用 @Order或者 @Priority 指定一下优先级,比如这样:

或者这样:

@Order/Priority  注解中值越小,那它的优先级就越高。

最后我们还可以将 PayService 接口所有实现类注入到 Map  中,其中里面的 key 就是Spring 的 Bean 的名字。

ps: 使用 @Autowired 注入属性到 Map 中,key 必须是 String 类型。

这也可以理解,你要是其他类型,Spring 都不知道去哪里给你找。。。

总结

@Autowired注解可以把相应 Bean 注入到相关属性中,它的注入方式有很多种。

我们可以使用 @Autowired直接标注类中的字段,快速使用。

我们也可以使用 @Autowired 标注构造方法/普通的方法,这样可以方便注入多个 Bean

最后,我们还可以 @Autowired标注 Array (数组),Collections(集合),甚至是 Map (散列表),将所有匹配到的 Bean 注入进来。

那最后一种方式,其实在某些场景非常有用。比如说,我们通过 @Autowired将所有匹配到 Bean 注入到 Map 中,利用这个特性,我们可以快速实现策略模式。

本文参考了原来@Autowired注解还可以这么玩? (qq.com)