欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring框架的ImportSelector詳細(xì)解讀

 更新時(shí)間:2024年01月29日 09:57:34   作者:Smallc0de  
這篇文章主要介紹了Spring框架的ImportSelector詳細(xì)解讀,Spring中一個(gè)非常重要的注解@Import中的ImportSelector接口的作用以及它到底有啥作用,也會(huì)捎帶一部分源碼說一下DeferredImportSelector是干啥的,需要的朋友可以參考下

前言

最近一直在鉆研Spring源碼,感覺要把自己看吐了,但是看到奧妙的地方還是會(huì)拍手稱快。

這次就總結(jié)一下Spring中一個(gè)非常重要的注解@Import中的ImportSelector接口的作用以及它到底有啥作用。

也會(huì)捎帶一部分源碼說一下DeferredImportSelector是干啥的,以及Spring解析這個(gè)和ImportSelector有什么區(qū)別。

ImportSelector

說到ImportSelector這個(gè)接口就不得不說這里面最重要的一個(gè)方法:selectImports()。

public interface ImportSelector {
   /**
    * Select and return the names of which class(es) should be imported based on
    * the {@link AnnotationMetadata} of the importing @{@link Configuration} class.
    * 選擇并返回需要導(dǎo)入的類的名稱,這些類基于AnnotationMetadata
    * 并且導(dǎo)入到@Configuration注解的類中的
    * @return the class names, or an empty array if none
    * 返回所有的class name,如果沒有,就返回空
    */
   String[] selectImports(AnnotationMetadata importingClassMetadata);
   /**
    * 返回排除的類,是一個(gè)類過濾器,但是這個(gè)方法被default注解了,
    * 可見Spring公司也知道,這個(gè)基本沒啥人用
    */
   @Nullable
   default Predicate<String> getExclusionFilter() {
      return null;
   }
}

源碼的注解里說了一大堆,就直接說這個(gè)方法能干啥吧。這個(gè)方法的返回值是一個(gè)字符串?dāng)?shù)組,只要在配置類被引用了,這里返回的字符串?dāng)?shù)組中的類名就會(huì)被Spring容器new出來,然后再把這些對(duì)象放到工廠當(dāng)中去。所以這有啥用呢?我們還是用一個(gè)例子演示一下。

ImportSelector簡單例子

首先我們先有一個(gè)實(shí)現(xiàn)了ImportSelector的類MyImportSelect,再構(gòu)造一個(gè)業(yè)務(wù)類IndexDao,然后配置類用@Import引入,最后測試類Test。

/**
 * 由于我們使用的ImportSelector所以就不需要放到Spring容器當(dāng)中了。
 * 我們要用@Import這個(gè)注解引入進(jìn)去。
 */
public class MyImportSelect implements ImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[]{IndexDao.class.getName()};
    }
}
@ComponentScan("com.demo")
@Import(MyImportSelect.class)
public class AppConfig {
}
public class IndexDao {
    public void query(){
        System.out.println("query IndexDao for MyImportSelect");
    }
}
public class Test {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext anno=new AnnotationConfigApplicationContext(AppConfig.class);
         anno.getBean(IndexDao.class).query();
         }
}
運(yùn)行打印
query IndexDao for MyImportSelect

從上面的例子看,盡管程序上沒有把MyImportSelect類放到Spring容器中,也沒有把IndexDao放到Spring容器中,但是在測試上就可以把IndexDao從容器中拿出來,并且正常執(zhí)行。

不知道大家看到這里有什么感覺,到這里我其實(shí)是有疑問的。我這么做有個(gè)卵用噻。我是有病吧,直接把類上加個(gè)@Component注冊(cè)進(jìn)去不香嗎?所以這個(gè)ImportSelector把程序搞這么復(fù)雜是有毛病吧,把簡單的功能搞這么復(fù)雜。

ImportSelector真正的作用

其實(shí)Spring公司既然這么設(shè)計(jì),那肯定是有用的。那么有什么用呢?設(shè)想這樣一個(gè)場景,如果有些功能我們并不需要Spring在一開始就加載進(jìn)去,而是需要Spring幫助我們把這些功能動(dòng)態(tài)加載進(jìn)去,這時(shí)候這個(gè)ImportSelector的作用就來了。我們完全可以把實(shí)現(xiàn)這個(gè)接口的類做成一個(gè)開關(guān),用來開啟或者關(guān)閉某一個(gè)或者某些功能類。

比如說我們上面的例子IndexDao,假設(shè)這個(gè)IndexDao實(shí)現(xiàn)的功能是一個(gè)擴(kuò)展功能,在正式的生產(chǎn)上不一定用的到。如果說一個(gè)包下有100個(gè)類,那么使用掃描去屏蔽這個(gè)類就很麻煩,但是屏蔽這個(gè)類用ImportSelector去做就很容易了。下次如果需要用到了,我再放開這個(gè)開關(guān),直接可以使用IndexDao的功能了,這樣就做到了一個(gè)靈活的功能掌控。

再舉一個(gè)實(shí)際的用例,假設(shè)我們的IndexDao不是打印而是返回一個(gè)針對(duì)代理IndexDao3的代理對(duì)象,比如輸出一個(gè)log。但是這個(gè)方法我不一定會(huì)用到,因?yàn)橹挥行枰玫酱淼臅r(shí)候,這個(gè)方法才有意義。我不需要去代理,這里的代碼就不要運(yùn)行。只有給一個(gè)顯示的通知,我這個(gè)代理才會(huì)去執(zhí)行。

public class IndexDao {
    public void query(){
	System.out.println("log for IndexDao3");
        return Proxy.newProxyInstance(IndexDao3);//偽碼
    }
}

看到這里,大家有沒有聯(lián)想到SpringAOP其實(shí)就是這個(gè)樣子。能夠做到動(dòng)態(tài)加載與卸載,與我們的程序沒有什么耦合關(guān)系。怎么才能實(shí)現(xiàn)這個(gè)所謂的動(dòng)態(tài)開啟呢?

ImportSelector開關(guān)

為了完成這個(gè)開關(guān),我們也模仿Spring寫一個(gè)EnableMySelector的注解,然后@Import我們自己的ImportSelector接口類。

@Retention(RetentionPolicy.RUNTIME) //開啟運(yùn)行時(shí)加載
@Import(MyImportSelect.class)
public @interface EnableMySelector {
}

做好了這一步在AppConfig這個(gè)配置里面就可以直接使用這個(gè)@ EnableMySelector自定義注解去開關(guān)一個(gè)類了。

@ComponentScan("com.demo")
@EnableMySelector
public class AppConfig { }
運(yùn)行,一樣打印
query IndexDao for MyImportSelect

講道理其實(shí)Spring中那么多的EnableXXXX的注解底層就是這樣的原理。到此還有誰敢說ImportSelector用處?。?/p>

連帶說一下DeferredImportSelector

這個(gè)是看Spring源碼的時(shí)候發(fā)現(xiàn)的,直接翻譯就是延時(shí)加載ImportSelector,實(shí)現(xiàn)這個(gè)接口的類,將會(huì)在@Configuration后面被加載,用法什么的和ImportSelector功能基本一樣。因?yàn)橛玫谋容^稀有就不多做解釋了,僅僅作為一個(gè)只是擴(kuò)展點(diǎn)介紹下。在ConfigurationClassParser中會(huì)有一個(gè)判斷,是不是這個(gè)接口,如果是就會(huì)放到后面解析。以下摘自源碼:

org.springframework.context.annotation.ConfigurationClassParser#processImports
//這里攔截了DeferredImportSelector然后使用handle()
if (selector instanceof DeferredImportSelector) {
   this.deferredImportSelectorHandler.handle(configClass, (DeferredImportSelector) selector);
}

進(jìn)入handle()方法,發(fā)現(xiàn)和這個(gè)接口相關(guān)的都被加入了一個(gè)deferredImportSelectors的list中。

public void handle(ConfigurationClass configClass, DeferredImportSelector importSelector) {
   DeferredImportSelectorHolder holder = new DeferredImportSelectorHolder(configClass, importSelector);
   if (this.deferredImportSelectors == null) {
      DeferredImportSelectorGroupingHandler handler = new DeferredImportSelectorGroupingHandler();
      handler.register(holder);
      handler.processGroupImports();
   }
   else {
	//加入到了一個(gè)ArrayList中
      this.deferredImportSelectors.add(holder);
   }
}

最終這個(gè)ArrayList在parse()方法的最后被處理了

org.springframework.context.annotation.ConfigurationClassParser#parse(java.util.Set<org.springframework.beans.factory.config.BeanDefinitionHolder>)
public void parse(Set<BeanDefinitionHolder> configCandidates) {
   //根據(jù)BeanDefinition的類型做不同的處理,一般都會(huì)調(diào)用ConfigurationClassParser.parse()進(jìn)行解析
   for (BeanDefinitionHolder holder : configCandidates) {
      BeanDefinition bd = holder.getBeanDefinition(); //拿出BeanDefinition
      try {
         if (bd instanceof AnnotatedBeanDefinition) {  //判斷是不是加了注解的
            // 解析注解對(duì)象,并且把解析出來的bd方法map中,但是這里的bd指的的普通的
            // 普通和不普通的怎么區(qū)分。比如@Bean和各種beanFactoryPostProcessor得到的bean
            //如果被加了注解,又調(diào)用了一個(gè)parse()方法
            parse(((AnnotatedBeanDefinition) bd).getMetadata(), holder.getBeanName());
         }
         else if (bd instanceof AbstractBeanDefinition && ((AbstractBeanDefinition) bd).hasBeanClass()) {
            parse(((AbstractBeanDefinition) bd).getBeanClass(), holder.getBeanName());
         }
         else {
            parse(bd.getBeanClassName(), holder.getBeanName());
         }
      }
      catch (BeanDefinitionStoreException ex) {
         throw ex;
      }
      catch (Throwable ex) {
         throw new BeanDefinitionStoreException(
               "Failed to parse configuration class [" + bd.getBeanClassName() + "]", ex);
      }
   }
	//處理,而此時(shí)上面其他的Import已經(jīng)處理完了
   this.deferredImportSelectorHandler.process();
}

到此這篇關(guān)于Spring框架的ImportSelector詳細(xì)解讀的文章就介紹到這了,更多相關(guān)ImportSelector詳細(xì)解讀內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • mybatis update set 多個(gè)字段實(shí)例

    mybatis update set 多個(gè)字段實(shí)例

    這篇文章主要介紹了mybatis update set 多個(gè)字段實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01
  • Spring事務(wù)相關(guān)問題解決方案

    Spring事務(wù)相關(guān)問題解決方案

    這篇文章主要介紹了Spring事務(wù)相關(guān)問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-02-02
  • Java之Algorithm_analysis案例詳解

    Java之Algorithm_analysis案例詳解

    這篇文章主要介紹了Java之Algorithm_analysis案例詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • Java方法的參數(shù)傳遞機(jī)制詳解

    Java方法的參數(shù)傳遞機(jī)制詳解

    這篇文章主要介紹了Java方法的參數(shù)傳遞機(jī)制詳解,對(duì)于Java初學(xué)者來說,剛學(xué)習(xí)Java的時(shí)候可能經(jīng)常會(huì)聽到調(diào)用方法時(shí)參數(shù)的值傳遞與引用傳遞,但是,實(shí)際上Java中方法的參數(shù)傳遞機(jī)制只有值傳遞,需要的朋友可以參考下
    2024-01-01
  • Spring整合Mybatis具體代碼實(shí)現(xiàn)流程

    Spring整合Mybatis具體代碼實(shí)現(xiàn)流程

    這篇文章主要介紹了Spring整合Mybatis實(shí)操分享,文章首先通過介紹Mybatis的工作原理展開Spring整合Mybatis的詳細(xì)內(nèi)容,需要的小伙伴可以參考一下
    2022-05-05
  • Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼

    Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼

    本篇文章主要介紹了Java實(shí)現(xiàn)迅雷地址轉(zhuǎn)成普通地址實(shí)例代碼,非常具有實(shí)用價(jià)值,有興趣的可以了解一下。
    2017-03-03
  • 注冊(cè)中心配置了spring?security后客戶端啟動(dòng)報(bào)錯(cuò)

    注冊(cè)中心配置了spring?security后客戶端啟動(dòng)報(bào)錯(cuò)

    這篇文章主要為大家介紹了注冊(cè)中心配置了spring?security后客戶端啟動(dòng)報(bào)錯(cuò)問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • Spring MVC 攔截器實(shí)現(xiàn)代碼

    Spring MVC 攔截器實(shí)現(xiàn)代碼

    本篇文章主要介紹了Spring MVC 攔截器的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2017-02-02
  • 最新評(píng)論