Spring源码学习:一篇搞懂@Autowire和@Resource注解的区别

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 最近在刷到很多文章讲解Spring IOC依赖注入时@Autowire和@Resource注解的区别,不同的文章总结出来的点有异同,所以还是看源码自己总结一下其两者的区别,及其用法。

前言

最近在刷到很多文章讲解Spring IOC依赖注入时@Autowire和@Resource注解的区别,不同的文章总结出来的点有异同,所以还是看源码自己总结一下其两者的区别,及其用法。

大家有没有想过两者的功能差不多,那为啥Spring还提供两种依赖注入方式呢?

我们知道@Autowire注解是Spring框架提供的,所以在使用该注解时依赖于该框架,而@Resource注解是JDK自带的,不需要依赖第三方。大多数IOC框架对@Resource都做了支持,而@Autowire只能使用Spring,所以当我们使用@Resource注解后,切换了其它框架,就可能不需要去修改该注解了,也能支持其注入功能。

正文

@Autowire注入过程

    public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {
        // 1、该方法会尝试从缓存中取当前Bean字段中使用了@Autowire的字段信息或方法信息。
        InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);
        try {
            // 进行属性注入
            metadata.inject(bean, beanName, pvs);
        }
        catch (BeanCreationException ex) {
            throw ex;
        }
        catch (Throwable ex) {
            throw new BeanCreationException(beanName, "Injection of autowired dependencies failed", ex);
        }
        return pvs;
    }

1、步骤1从缓存中取不到值时,会通过反射的方式遍历当前bean的所有字段信息和方法信息,并判断其是否使用了@Autowire注解,并封装到对应的处理器中,后面注入步骤会进行遍历进行查询处理及注入。@Autowire可以注解在方法、字段、构造器上,构造器注入在创建bean的时候进行的。

2、步骤2会根据步骤1中查询出来的信息遍历,由于字段和方法会分别封装成InjectedElement类型的不同子类实现,所以不同的类型会调用不同的子类实现方法进行处理。

doResolveDependency

    @Nullable
    public Object doResolveDependency(DependencyDescriptor descriptor, @Nullable String beanName,
            @Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException {
            //..................
            //此处省略了部分代码
            //..................

            //尝试从bean工厂中查询出需要注入类型的所有Bean对象,@Qualifier注解的解析是在这个步骤中进行的
            Map<String, Object> matchingBeans = findAutowireCandidates(beanName, type, descriptor);
            //beanFactory工厂中查询不到时,抛出异常
            if (matchingBeans.isEmpty()) {
                //如果descriptor需要注入
                if (isRequired(descriptor)) {
                    //抛出NoSuchBeanDefinitionException或BeanNotOfRequiredTypeException以解决不可 解决的依赖关系
                    raiseNoMatchingBeanFound(type, descriptor.getResolvableType(), descriptor);
                }
                //返回null,表示么有找到候选Bean对象
                return null;
            }

            //定义用于存储唯一的候选Bean名变量
            String autowiredBeanName;
            //定义用于存储唯一的候选Bean对象变量
            Object instanceCandidate;

            //如果beanFactory工厂中存在多个类型的bean时
            if (matchingBeans.size() > 1) {
                //筛选出符合要求的bean
                autowiredBeanName = determineAutowireCandidate(matchingBeans, descriptor);
                //如果autowiredBeanName为null
                if (autowiredBeanName == null) {
                    //如果查询不到合适的,而且该属性并非require=true的,可以赋值为null
                    if (isRequired(descriptor) || !indicatesMultipleBeans(type)) {
                        //让descriptor尝试选择其中一个实例,默认实现是抛出NoUniqueBeanDefinitionException.
                        return descriptor.resolveNotUnique(descriptor.getResolvableType(), matchingBeans);
                    }
                    else {
                        // In case of an optional Collection/Map, silently ignore a non-unique case:
                        // possibly it was meant to be an empty collection of multiple regular beans
                        // (before 4.3 in particular when we didn't even look for collection beans).
                        // 如果是可选的Collection/Map,则静默忽略一个非唯一情况:
                        // 可能是多个常规bean的空集合
                        // (尤其是在4.3之前,设置在我们没有寻找collection bean的时候 )
                        return null;
                    }
                }
                //获取autowiredBeanName对应的候选Bean对象
                instanceCandidate = matchingBeans.get(autowiredBeanName);
            }
            else {
                //如果查询出来只有一个时,则将该bean作为注入值
                Map.Entry<String, Object> entry = matchingBeans.entrySet().iterator().next();
                //让autowireBeanName引用该元素的候选bean名
                autowiredBeanName = entry.getKey();
                //让instanceCandidate引用该元素的候选bean对象
                instanceCandidate = entry.getValue();
            }

            //如果候选bean名不为null,
            if (autowiredBeanNames != null) {
                //将autowiredBeanName添加到autowiredBeanNames中,又添加一次
                autowiredBeanNames.add(autowiredBeanName);
            }
            //如果instanceCandidate是Class实例,对其进行实例化
            if (instanceCandidate instanceof Class) {
                //让instanceCandidate引用 descriptor对autowiredBeanName解析为该工厂的Bean实例
                instanceCandidate = descriptor.resolveCandidate(autowiredBeanName, type, this);
            }
            //定义一个result变量,用于存储最佳候选Bean对象
            Object result = instanceCandidate;
            //如果reuslt是NullBean的实例
            if (result instanceof NullBean) {
                //如果descriptor需要注入
                if (isRequired(descriptor)) {
                    //抛出NoSuchBeanDefinitionException或BeanNotOfRequiredTypeException以解决不可 解决的依赖关系
                    raiseNoMatchingBeanFound(type, descriptor.getResolvableType(), descriptor);
                }
                //返回null,表示找不到最佳候选Bean对象
                result = null;
            }
            //如果result不是type的实例
            if (!ClassUtils.isAssignableValue(type, result)) {
                //抛出Bean不是必需类型异常
                throw new BeanNotOfRequiredTypeException(autowiredBeanName, type, instanceCandidate.getClass());
            }
            //返回最佳候选Bean对象【result】
            return result;
        }
        finally {
            //设置上一个切入点对象
            ConstructorResolver.setCurrentInjectionPoint(previousInjectionPoint);
        }
    }

1、尝试从缓存中获取被注入类型的所有Bean;
2、并对获取到Bean容器进行遍历,判断被注入类型是否有@Qualifier注解,有则进行名称匹配,匹配到时返回一个符合条件的Bean容器,否则返回所有注入类型的Bean用于后续进一步筛选。
3、如果获取不到时,判断该属性是否必须注入,如果非必须注入可以注入null;
4、如果匹配到的类型Bean只有一个时,则认为是符合要求的,返回该值后进行注入。
5、如果匹配到多个Bean时,根据匹配条件来筛选;

@Qualifier匹配过程见: isAutowireCandidate(BeanDefinitionHolder bdHolder, DependencyDescriptor descriptor)方法详解

多个bean的匹配过程见: determineAutowireCandidate(matchingBeans, descriptor)方法详解

isAutowireCandidate

    public boolean isAutowireCandidate(BeanDefinitionHolder bdHolder, DependencyDescriptor descriptor) {
        //判断要注入的字段是否有@Quilifier注解
        boolean match = super.isAutowireCandidate(bdHolder, descriptor);
        if (match) {
            //校验是否匹配
            match = checkQualifiers(bdHolder, descriptor.getAnnotations());
            if (match) {
                //如果是方法注入时,获取方法参数进行校验匹配
                MethodParameter methodParam = descriptor.getMethodParameter();
                if (methodParam != null) {
                    Method method = methodParam.getMethod();
                    if (method == null || void.class == method.getReturnType()) {
                        match = checkQualifiers(bdHolder, methodParam.getMethodAnnotations());
                    }
                }
            }
        }
        return match;
    }

determineAutowireCandidate

    protected String determineAutowireCandidate(Map<String, Object> candidates, DependencyDescriptor descriptor) {
        //通过反射的方式去查询已匹配的Bean容器中是否有@Primary注解,如果有多个则抛出异常,如果只有一个则代表该Bean对象为符合注入条件的
        Class<?> requiredType = descriptor.getDependencyType();
        String primaryCandidate = determinePrimaryCandidate(candidates, requiredType);
        if (primaryCandidate != null) {
            return primaryCandidate;
        }
        //判断已匹配的Bean容器中是否有@Priority注解,并比较其优先级大小,挑选出符合的一个
        String priorityCandidate = determineHighestPriorityCandidate(candidates, requiredType);
        if (priorityCandidate != null) {
            return priorityCandidate;
        }
        // Fallback
        // 这里将匹配的bean容器遍历后,挨个判断bean名称与要注入类型的名称是否相同,不同则去别名缓存中查询,看是否有别名与要注入类型的名称一样,有则匹配
        for (Map.Entry<String, Object> entry : candidates.entrySet()) {
            String candidateName = entry.getKey();
            Object beanInstance = entry.getValue();
            if ((beanInstance != null && this.resolvableDependencies.containsValue(beanInstance)) ||
                    matchesBeanName(candidateName, descriptor.getDependencyName())) {
                return candidateName;
            }
        }
        return null;
    }

1、通过反射的方式去查询已匹配的Bean容器中是否有@Primary注解,如果有多个则抛出异常,如果只有一个则代表该Bean对象为符合注入条件的

2、判断已匹配的Bean容器中是否有@Priority注解,并比较其优先级大小,挑选出符合的一个

3、这里将匹配的bean容器遍历后,挨个判断bean名称与要注入类型的名称是否相同,不同则去别名缓存中查询,看是否有别名与要注入类型的名称一样,有则匹配;

@Resource注入过程

    public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {
        //查询注入对象或者方法中是否有@Resource注解
        InjectionMetadata metadata = findResourceMetadata(beanName, bean.getClass(), pvs);
        try {
            //对其进行解析注入
            metadata.inject(bean, beanName, pvs);
        }
        catch (Throwable ex) {
            throw new BeanCreationException(beanName, "Injection of resource dependencies failed", ex);
        }
        return pvs;
    }

1、遍历当前Bean的属性及其方法,判断是否有@Resource注解,并封装成对应的包装类

2、对其进行查找注入

autowireResource

protected Object autowireResource(BeanFactory factory, LookupElement element, @Nullable String requestingBeanName)
            throws NoSuchBeanDefinitionException {

        // 自动装配的对象
        Object resource;
        // 自动装配的名字
        Set<String> autowiredBeanNames;
        // 依赖的属性名
        String name = element.name;
        //默认BeanFacory是AutowireCapableBeanFactory工厂类型
        if (factory instanceof AutowireCapableBeanFactory) {
            AutowireCapableBeanFactory beanFactory = (AutowireCapableBeanFactory) factory;
            // 创建依赖描述
            DependencyDescriptor descriptor = element.getDependencyDescriptor();
            //这里的factory.containsBean(name)会根据需要注入的属性名称去BeanFactory工厂中进行查询,如果查询到了直接获取后进行注入,如果查询不到时,调用beanFactory.resolveDependency方法进行查找,该方法流程与@Autowire一样。
            if (this.fallbackToDefaultTypeMatch && element.isDefaultName && !factory.containsBean(name)) {
                //如果容器中还没有此bean,则会使用resolveDependency()方法将符合bean type的bean definetion调用一次getBean()
                // 从这些bean选出符合requestingBeanName的bean
                autowiredBeanNames = new LinkedHashSet<>();
                resource = beanFactory.resolveDependency(descriptor, requestingBeanName, autowiredBeanNames, null);
                if (resource == null) {
                    throw new NoSuchBeanDefinitionException(element.getLookupType(), "No resolvable resource object");
                }
            }
            else {
                //如果容器中有此bean则取出这个bean对象作为属性值
                resource = beanFactory.resolveBeanByName(name, descriptor);
                autowiredBeanNames = Collections.singleton(name);
            }
        }
        else {
            resource = factory.getBean(name, element.lookupType);
            autowiredBeanNames = Collections.singleton(name);
        }

        //.......
        //.......

        return resource;
    }

@Resouce解析与@Autowire解析异同:

1、@Resouce会先尝试使用beanName名称去beanFactory工厂中查询是否有该定义信息,如果有则直接取出。

2、如果步骤1不满足时,走后面按类型匹配的步骤,该步骤与@Autowire调用的方法一致,所以流程是一样的;

总结

@Autowire解析步骤:

  1. 尝试从缓存中获取被注入类型的所有Bean;
  2. 并对获取到Bean容器进行遍历,判断被注入类型是否有@Qualifier注解,有则进行名称匹配,匹配到时返回一个符合条件的Bean容器,否则返回所有注入类型的Bean用于后续进一步筛选。
  3. 如果获取不到时,判断该属性是否必须注入,如果非必须注入可以注入null;
  4. 如果匹配到的类型Bean只有一个时,则认为是符合要求的,返回该值后进行注入。
  5. 如果匹配到多个Bean时,根据匹配条件来筛选:

    • 通过反射的方式去查询已匹配的Bean容器中是否有@Primary注解,如果有多个则抛出异常,如果只有一个则代表该Bean对象为符合注入条件的
    • 判断已匹配的Bean容器中是否有@Priority注解,并比较其优先级大小,挑选出符合的一个,数值越低优选级越高
    • 这里将匹配的bean容器遍历后,挨个判断bean名称与要注入类型的名称是否相同,不同则去别名缓存中查询,看是否有别名与要注入类型的名称一样,有则匹配;

@Resouce解析步骤:

  1. 根据beanName去工厂中查询是否有该定义信息,有则获取Bean对象,没有则进行类型匹配操作;
  2. 尝试从缓存中获取被注入类型的所有Bean;
  3. 并对获取到Bean容器进行遍历,判断被注入类型是否有@Qualifier注解,有则进行名称匹配,匹配到时返回一个符合条件的Bean容器,否则返回所有注入类型的Bean用于后续进一步筛选。
  4. 如果获取不到时,判断该属性是否必须注入,如果非必须注入可以注入null;
  5. 如果匹配到的类型Bean只有一个时,则认为是符合要求的,返回该值后进行注入。
  6. 如果匹配到多个Bean时,根据匹配条件来筛选:

    • 通过反射的方式去查询已匹配的Bean容器中是否有@Primary注解,如果有多个则抛出异常,如果只有一个则代表该Bean对象为符合注入条件的;
    • 判断已匹配的Bean容器中是否有@Priority注解,并比较其优先级大小,挑选出符合的一个,数值越低优选级越高;
    • 这里将匹配的bean容器遍历后,挨个判断bean名称与要注入类型的名称是否相同,不同则去别名缓存中查询,看是否有别名与要注入类型的名称一样,有则匹配;
目录
相关文章
|
2月前
|
XML Java 数据格式
SpringBoot入门(8) - 开发中还有哪些常用注解
SpringBoot入门(8) - 开发中还有哪些常用注解
61 0
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
110 2
|
26天前
|
Java Spring
【Spring】方法注解@Bean,配置类扫描路径
@Bean方法注解,如何在同一个类下面定义多个Bean对象,配置扫描路径
168 73
|
21天前
|
Java Spring 容器
【SpringFramework】Spring IoC-基于注解的实现
本文主要记录基于Spring注解实现IoC容器和DI相关知识。
50 21
|
13天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
26天前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
|
26天前
|
Java Spring
【Spring配置】idea编码格式导致注解汉字无法保存
问题一:对于同一个项目,我们在使用idea的过程中,使用汉字注解完后,再打开该项目,汉字变成乱码问题二:本来a项目中,汉字注解调试好了,没有乱码了,但是创建出来的新的项目,写的注解又成乱码了。
|
1月前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
65 2
|
1月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
2月前
|
前端开发 Java Spring
Spring MVC核心:深入理解@RequestMapping注解
在Spring MVC框架中,`@RequestMapping`注解是实现请求映射的核心,它将HTTP请求映射到控制器的处理方法上。本文将深入探讨`@RequestMapping`注解的各个方面,包括其注解的使用方法、如何与Spring MVC的其他组件协同工作,以及在实际开发中的应用案例。
55 4