Spring在單值注入時如何按類型查找匹配的Bean,相信很多沒有經(jīng)驗(yàn)的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
創(chuàng)新互聯(lián)建站主營廣德網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營網(wǎng)站建設(shè)方案,App定制開發(fā),廣德h5微信小程序開發(fā)搭建,廣德網(wǎng)站營銷推廣歡迎廣德等地區(qū)企業(yè)咨詢
我經(jīng)常寫如下代碼:
@Autowired private AService aservice;
不知你是否也好奇,Spring是如何找到AService類型的Bean的呢?,我們就聊聊這個->單值注入時如何按類型查找匹配的Bean.
很簡單,核心就3步。
Spring在DefaultListableBeanFactory.findAutowireCandidates方法中實(shí)現(xiàn)。 其部分源碼如下:
String[] candidateNames = BeanFactoryUtils .beanNamesForTypeIncludingAncestors ( this, requiredType, true, descriptor.isEager());
這個beanNamesForTypeIncludingAncestors的作用就是,獲取requiredType(AService)類型所有匹配的beanName(包含先祖BeanFactory)。
beanNamesForTypeIncludingAncestors內(nèi)部是如果實(shí)現(xiàn)的呢?我概括了下簡要邏輯如下:
遍歷所有的BeanDefinition,獲得所有的BeanName.
針對所有的BeanName,先嘗試獲取單例進(jìn)行匹配,若未匹配上再以Bean Definition進(jìn)行匹配。
匹配時,如果Bean是FactoryBean,先嘗試FactoryBean生產(chǎn)的實(shí)際Bean進(jìn)行匹配,若未匹配上再以FactoryBean 進(jìn)行匹配。
DefaultListableBeanFactory.determinePrimaryCandidate實(shí)現(xiàn)了篩選首選Bean的邏輯, 其中的核心方法是isPrimary,該方法是判斷當(dāng)前Bean是否是首選Bean的。源碼如下:
protected boolean isPrimary(String beanName, Object beanInstance) { if (containsBeanDefinition(beanName)) { return getMergedLocalBeanDefinition(beanName).isPrimary(); } BeanFactory parent = getParentBeanFactory(); return (parent instanceof DefaultListableBeanFactory && ((DefaultListableBeanFactory) parent).isPrimary(beanName,beanInstance)); }
getMergedLocalBeanDefinition(beanName).isPrimary()方法,對應(yīng)AbstractBeanDefinition的primary屬性,該屬性被賦值的地方是在AnnotatedBeanDefinitionReader.doRegisterBean方法中。有如下邏輯。
//省略甚多代碼...... for (Class extends Annotation> qualifier : qualifiers) { if (Primary.class == qualifier) { abd.setPrimary(true); } //省略很多代碼....
看到這,我們可以得出一個結(jié)論:
被@Primary注解的bean,單值注入時會作為首選。
Spring是如何確定Bean的優(yōu)先級的呢?
在DefaultListableBeanFactory.determineHighestPriorityCandidate中,實(shí)現(xiàn)按優(yōu)先級選擇Bean其中,獲取Bean的優(yōu)先級的邏輯在getPriority方法中,如下:
protected Integer getPriority(Object beanInstance) { Comparator
查看OrderComparator的實(shí)現(xiàn)類AnnotationAwareOrderComparator中的源碼發(fā)現(xiàn), 獲取優(yōu)先級的邏輯實(shí)際在在OrderUtils.getPriority 中
public static Integer getPriority(Class> type) { if (priorityAnnotationType == null) { return null; } Object cached = priorityCache.get(type); if (cached != null) { return (cached instanceof Integer ? (Integer) cached : null); } Annotation priority = AnnotationUtils.findAnnotation(type, priorityAnnotationType); Integer result = null; if (priority != null) { result = (Integer) AnnotationUtils.getValue(priority); } priorityCache.put(type, (result != null ? result : NOT_ANNOTATED)); return result; }
在OrderUtils 向上查找發(fā)現(xiàn) priorityAnnotationType的值為:
priorityAnnotationType = (Class extends Annotation>) ClassUtils.forName("javax.annotation.Priority", OrderUtils.class.getClassLoader());
被@Priority注解的類,其值越小,在單值注入時,越優(yōu)先選擇。
Spring的源碼非常多,僅有這3步當(dāng)然是不行的,我準(zhǔn)備了流程圖,梳理了Spring單值注入時查找匹配Bean的流程。
看完上述內(nèi)容,你們掌握Spring在單值注入時如何按類型查找匹配的Bean的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!