当前位置:首页 > 科技  > 软件

一个很有意思的Spring注入问题,你遇到过吗?

来源: 责编: 时间:2024-03-18 09:36:48 338观看
导读环境:Spring5.3.231. 问题描述static interface DAO {}static class CommonDAO implements DAO {}@Configurationstatic class AppConfig { @Bean DAO dao() { return new CommonDAO() ; }}static class CommonS

环境:Spring5.3.23AKJ28资讯网——每日最新资讯28at.com

1. 问题描述

static interface DAO {}static class CommonDAO implements DAO {}@Configurationstatic class AppConfig {  @Bean  DAO dao() {    return new CommonDAO() ;  }}static class CommonService {  @Resource  private DAO dao ;  @Resource  private CommonDAO commonDAO ;}try (AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext()) {  context.registerBean(AppConfig.class) ;  context.registerBean(CommonService.class) ;  context.refresh() ;}

上面是基本的bean定义。在AppConfig配置类中定义了DAO bean实例,在CommonService中分别去注入DAO 接口和CommonDAO。运行上面的程序没有问题正常。AKJ28资讯网——每日最新资讯28at.com

2. 问题汇总

2.1 修改注入1

static class CommonService {  @Resource  private CommonDAO commonDAO ;}

当CommonService只注入CommonDAO时,程序既然报错了AKJ28资讯网——每日最新资讯28at.com

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.pack.main.bean_propertyvalue_inject.InterfaceAndImplInject$CommonDAO' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@javax.annotation.Resource(shareable=true, lookup=, name=, description=, authenticationType=CONTAINER, type=class java.lang.Object, mappedName=)}  at org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoMatchingBeanFound(DefaultListableBeanFactory.java:1801)  at org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:1357)

错误提示:需要CommonDAO但是容器中没有,是不是很奇怪。AKJ28资讯网——每日最新资讯28at.com

2.2. 修改注入2

static class CommonService {  @Resource  private CommonDAO dao;}

只是吧字段的名称修改为dao,程序又正确了。这个什么原因???AKJ28资讯网——每日最新资讯28at.com

2.3 修改注入3

static class CommonService {  @Resource  private CommonDAO commonDAO ;  @Resource  private DAO dao ;}

这里仅仅是修改了下字段的顺序,程序又报错了,是不是太神奇了。AKJ28资讯网——每日最新资讯28at.com

2.4 修改注入4

@Configurationstatic class AppConfig {  @Bean  CommonService commonService() {    return new CommonService() ;  }  @Bean  DAO dao() {    return new CommonDAO() ;  }}static class CommonService {  @Resource  private CommonDAO commonDAO ;}

修改了CommonService bean的注册方式,运行程序还是错误AKJ28资讯网——每日最新资讯28at.com

2.5 修改注入5

@Configurationstatic class AppConfig {  @Bean  DAO dao() {    return new CommonDAO() ;  }  @Bean  CommonService commonService() {    return new CommonService() ;  }}

根据2.4的情况,修改注册DAO与CommonService的顺序后,程序又正确了。AKJ28资讯网——每日最新资讯28at.com

3. 原因解析

当如下方式注入时AKJ28资讯网——每日最新资讯28at.com

@Resourceprivate DAO dao ;@Resourceprivate CommonDAO commonDAO ;

提示:@Resource注解对应的处理器是CommonAnnotationBeanPostProcessorAKJ28资讯网——每日最新资讯28at.com

这里首先要整清楚@Resource的注入方式AKJ28资讯网——每日最新资讯28at.com

@Resource先根据beanName进行查找,再通过类型查找。源码:AKJ28资讯网——每日最新资讯28at.com

public class CommonAnnotationBeanPostProcessor {  protected Object autowireResource(BeanFactory factory, LookupElement element, @Nullable String requestingBeanName) {    Object resource;    if (factory instanceof AutowireCapableBeanFactory) {      AutowireCapableBeanFactory beanFactory = (AutowireCapableBeanFactory) factory;      DependencyDescriptor descriptor = element.getDependencyDescriptor();      // 判断你当前注入属性的名字(beanName) 在容器中是否存在。这里取反了,如果不存在时进行类型的查找      if (this.fallbackToDefaultTypeMatch && element.isDefaultName && !factory.containsBean(name)) {        resource = beanFactory.resolveDependency(descriptor, requestingBeanName, autowiredBeanNames, null);      } else {        // 存在,直接通过beanName(这里就是字段名)查找        resource = beanFactory.resolveBeanByName(name, descriptor);        autowiredBeanNames = Collections.singleton(name);      }    }    return resource;  }}

上面你知道了@Resource注解的方式注入的方式后。接下来就是查找具体的bean了,不管是通过beanName还是类型。这里演示还是按照beanName方式,接着上面的代码AKJ28资讯网——每日最新资讯28at.com

public abstract class AbstractAutowireCapableBeanFactory {  public Object resolveBeanByName(String name, DependencyDescriptor descriptor) {    return getBean(name, descriptor.getDependencyType());  }}public abstract class AbstractBeanFactory {  public <T> T getBean(String name, Class<T> requiredType) throws BeansException {    return doGetBean(name, requiredType, null, false);  }  protected <T> T doGetBean(    String name, @Nullable Class<T> requiredType, @Nullable Object[] args, boolean typeCheckOnly) {    // 这里就是先从单例池中获取指定beanName是否存在,如果不存在则进行创建bean实例。    // 创建完成后将当前的实例存入单例池中。  }}

到此,DAO类型的属性就注入成功了,接下是注入CommonDAO。注入CommonDAO由于容器中没有对应的beanName,所有进入上面的if语句中。AKJ28资讯网——每日最新资讯28at.com

public class DefaultListableBeanFactory {  public Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName,    @Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException {    // ...    Object result = doResolveDependency(descriptor, requestingBeanName, autowiredBeanNames, typeConverter);  }  public Object doResolveDependency(...) {    // ...    Map<String, Object> matchingBeans = findAutowireCandidates(beanName, type, descriptor);  }  protected Map<String, Object> findAutowireCandidates(    @Nullable String beanName, Class<?> requiredType, DependencyDescriptor descriptor) {    // 通过类型查找beanNames, 当前reqiredType=CommonDAO    String[] candidateNames = BeanFactoryUtils.beanNamesForTypeIncludingAncestors(        this, requiredType, true, descriptor.isEager());  }}public abstract class BeanFactoryUtils {  public static String[] beanNamesForTypeIncludingAncestors(      ListableBeanFactory lbf, Class<?> type, boolean includeNonSingletons, boolean allowEagerInit) {    // 通过类型查找    String[] result = lbf.getBeanNamesForType(type, includeNonSingletons, allowEagerInit);    return result;  }}public class DefaultListableBeanFactory {  public String[] getBeanNamesForType(@Nullable Class<?> type, boolean includeNonSingletons, boolean allowEagerInit) {    // 通过类型查找    String[] resolvedBeanNames = doGetBeanNamesForType(ResolvableType.forRawClass(type), includeNonSingletons, true);    return resolvedBeanNames;  }  private String[] doGetBeanNamesForType(ResolvableType type, boolean includeNonSingletons, boolean allowEagerInit) {    // 遍历所有的BeanDefinition(这是Spring容器对每一个bena的元数据了)    for (String beanName : this.beanDefinitionNames) {      RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName) ;      // 关键代码      matchFound = isTypeMatch(beanName, type, true);    }  }  protected boolean isTypeMatch(String name, ...) {    // beanName = dao    String beanName = transformedBeanName(name);    // 从单例池中获取实例,这里肯定可以获取,我们第一个属性注入的就是    // DAO,所以这里就返回了CommonDAO实例    Object beanInstance = getSingleton(beanName, false);    if (beanInstance != null && beanInstance.getClass() != NullBean.class) {      // 这里肯定是实例对象,直接返回了      if (typeToMatch.isInstance(beanInstance)) {        return true;      }    }  }}

到这你应该清楚了为什么同时有DAO和CommonDAO注入时能成功了。但是当没有DAO注入的时候为什么就错误呢?原因其实在上面已经给出了,你只要包装我在注入CommonDAO时,容器中已经将DAO这个bean实例创建存入到单例池中即可。这也就是为什么上面我们调整合理的顺序后就能注入成功。还有就是你可以将CommonDAO的字段名称改成与DAO bean的beanName一致也是可以的。AKJ28资讯网——每日最新资讯28at.com

以上是本篇文章的全部内容,希望对你有帮助。AKJ28资讯网——每日最新资讯28at.com

完毕!!!AKJ28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-76486-0.html一个很有意思的Spring注入问题,你遇到过吗?

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 没看过ReentrantLock源码,别说精通Java并发编程

下一篇: JQuery 4.0震撼发布:这是复兴还是告别?

标签:
  • 热门焦点
  • 影音体验是真的强 简单聊聊iQOO Pad

    大公司的好处就是产品线丰富,非常细分化的东西也能给你做出来,例如早先我们看到了新的vivo Pad2,之后我们又在iQOO Neo8 Pro的发布会上看到了iQOO的首款平板产品iQOO Pad。虽
  • 6月安卓手机性价比榜:Note 12 Turbo断层式碾压

    6月份有一个618,虽然这是京东周年庆的日子,但别的电商也都不约而同的跟进了,反正促销没坏处,厂商和用户都能满意。618期间一些产品也出现了历史低价,那么各个价位段的产品性价比
  • Automa-通过连接块来自动化你的浏览器

    1、前言通过浏览器插件可实现自动化脚本的录制与编写,具有代表性的工具就是:Selenium IDE、Katalon Recorder,对于简单的业务来说可快速实现自动化的上手工作。Selenium IDEKat
  • 谷歌KDD'23工作:如何提升推荐系统Ranking模型训练稳定性

    谷歌在KDD 2023发表了一篇工作,探索了推荐系统ranking模型的训练稳定性问题,分析了造成训练稳定性存在问题的潜在原因,以及现有的一些提升模型稳定性方法的不足,并提出了一种新
  • 一个注解实现接口幂等,这样才优雅!

    场景码猿慢病云管理系统中其实高并发的场景不是很多,没有必要每个接口都去考虑并发高的场景,比如添加住院患者的这个接口,具体的业务代码就不贴了,业务伪代码如下:图片上述代码有
  • 花7万退货退款无门:谁在纵容淘宝珠宝商家造假?

    来源:极点商业作者:杨铭在淘宝购买珠宝玉石后,因为保证金不够赔付,店铺关闭,退货退款难、维权无门的比比皆是。&ldquo;提供相关产品鉴定证书,支持全国复检,可以30天无理由退换货。&
  • 一条抖音4亿人围观 ! 这家MCN比无忧传媒还野

    作者:Hiu 来源:互联网品牌官01 擦边少女空降热搜,幕后推手曝光被网友誉为&ldquo;纯欲天花板&rdquo;的女网红井川里予,近期因为一组哥特风照片登上热搜,引发了一场互联网世界关于
  • 2纳米决战2025

    集微网报道 从三强争霸到四雄逐鹿,2nm的厮杀声已然隐约传来。无论是老牌劲旅台积电、三星,还是誓言重回先进制程领先地位的英特尔,甚至初成立不久的新
  • 2299元起!iQOO Pad明晚首销:性能最强天玑平板

    5月23日,iQOO如期举行了新品发布会,除了首发安卓最强旗舰处理器的iQOO Neo8系列新机外,还在发布会上推出了旗下首款平板电脑——iQOO Pad,其最大的卖点
Top