Spring自定義擴(kuò)展

借助@Component

Spring本身的@Component實(shí)現(xiàn),依賴于ClassPathScanningCandidateComponentProvider在指定的路徑下進(jìn)行掃描,并且完成自動(dòng)加載。
查看ClassPathScanningCandidateComponentProvider的源碼,可以看到Spring默認(rèn)加載了一個(gè)注解過(guò)濾器,來(lái)過(guò)濾@Component的類(lèi)。

protected void registerDefaultFilters() {   
   this.includeFilters.add(new AnnotationTypeFilter(Component.class));
   ClassLoader cl = ClassPathScanningCandidateComponentProvider.class.getClassLoader();
   try {    
      this.includeFilters.add(new AnnotationTypeFilter(((Class<? extends Annotation>) ClassUtils.forName("javax.annotation.ManagedBean", cl)), false)); 
      logger.debug("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning"); 
   }   catch (ClassNotFoundException ex) {     
     // JSR-250 1.1 API (as included in Java EE 6) not available - simply skip.   
   }   
   try {      
      this.includeFilters.add(new AnnotationTypeFilter(((Class<? extends Annotation>) ClassUtils.forName("javax.inject.Named", cl)), false));      
      logger.debug("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");   
   }  
   catch (ClassNotFoundException ex) {     
   // JSR-330 API not available - simply skip.  
   }
}

這里只加載了@Component注解的過(guò)濾器,那@Service、@Controller這些注解呢?看了下@Service的源碼,就知道了@Service其實(shí)是借助了@Component來(lái)實(shí)現(xiàn)加載的。

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Service {

    /**
     * The value may indicate a suggestion for a logical component name,
     * to be turned into a Spring bean in case of an autodetected component.
     * @return the suggested component name, if any
     */
    String value() default "";

}

這種方式比較局限,不能做一些定制。

Spring啟動(dòng)之后,再把bean拿出來(lái)加工、定制

SpringMVC就是使用這種方式,在Spring啟動(dòng)之后,遍歷所有的bean,把帶有@Controller的bean拿出來(lái),構(gòu)造UrlMapping。比如AbstractDetectingUrlHandlerMapping 。

自定義掃描

BeanFactoryPostProcessor 和ApplicationContextAware。
Spring提供了一些的接口使程序可以嵌入Spring的加載過(guò)程。這個(gè)類(lèi)中的繼承ApplicationContextAware接口,Spring會(huì)讀取ApplicationContextAware類(lèi)型的的JavaBean,并調(diào)用setApplicationContext(ApplicationContext applicationContext)傳入Spring的applicationContext。
同樣繼承BeanFactoryPostProcessor接口,Spring會(huì)在BeanFactory的相關(guān)處理完成后調(diào)用postProcessBeanFactory方法,進(jìn)行定制的功能。

ClassPathBeanDefinitionScanner是Spring提供的一個(gè)ClassPath掃描器,其實(shí)也繼承于ClassPathScanningCandidateComponentProvider,只不過(guò)ClassPathBeanDefinitionScanner需要一個(gè)BeanFactory或者ApplicationContext,在掃描到符合要求的類(lèi),就會(huì)加入到BeanFactory或者ApplicationContext中。

至此,完成了自定義掃描功能,但是還沒(méi)有對(duì)bean進(jìn)行加工、定制,還需要使用到FactoryBean,普通的JavaBean是直接使用類(lèi)的實(shí)例,但是如果一個(gè)Bean繼承了這個(gè)借口,就可以通過(guò)getObject()方法來(lái)自定義實(shí)例的內(nèi)容,在FactoryBeanTest的getObject()就通過(guò)代理了原始類(lèi)的方法,自定義類(lèi)的方法。

總結(jié)一下:

  • ApplicationContextAware獲取ApplicationContext
  • BeanFactoryPostProcessor 獲取BeanFactory
  • ClassPathBeanDefinitionScanner提供了自定義掃描的入口(使用ApplicationContext和BeanFactory,并且啟動(dòng)自定義掃描)
  • FactoryBean對(duì)自己感興趣的Bean進(jìn)行增強(qiáng)處理
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,715評(píng)論 19 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 47,290評(píng)論 6 342
  • 1- IOC的概念 IOC:也即控制反轉(zhuǎn),DI即依賴注入,控制反轉(zhuǎn)IOC和依賴注入DI其實(shí)就是同個(gè)概念的兩個(gè)不同...
    zhanglbjames閱讀 3,149評(píng)論 1 3
  • 畢業(yè)之后,才明白別人嘴里說(shuō)的大學(xué)生活多美好,每當(dāng)工作不順心時(shí),總是會(huì)想起在大學(xué)的快樂(lè)時(shí)光。那些陪你一起瘋一起笑的朋...
    鯨與樹(shù)閱讀 826評(píng)論 3 6
  • 精進(jìn):如何成為一個(gè)厲害的人 1.時(shí)間分配的重心為長(zhǎng)半衰,當(dāng)下收益-半衰期圖。當(dāng)下好處的半衰期短,則只有通過(guò)持續(xù)做而...
    hanalt閱讀 210評(píng)論 0 0

友情鏈接更多精彩內(nèi)容