快訊 >

        世界觀天下!【Spring源碼】- 07 擴展點之自定義標簽

        時間:2023-03-29 09:30:05       來源:騰訊云

        Spring中正逐漸采用注解方式取代XML配置方式,所以,使用XML配置的機會正越來越少。然后,如果你開發的工具模塊可能會被很多系統使用,考慮到兼容性問題,就需要提供XML方式集成,這時就需要自定義標簽;還有,你在看一些開源源碼時,一般也是提供自定義標簽方式集成。所以,還是可以去了解一下自定義標簽實現。

        Spring中使用自定義標簽還是比較簡單,下面我們就實現一個自定義標簽,其功能類似標簽:將指定包路徑下帶有指定注解的Bean掃描注冊。


        【資料圖】

        1、首先,在resources/META-INF目錄下定義一個xsd文件,描述自定義標簽屬性:

                                                                                        

        2、自定義NamespaceHandler,注冊使用CustomScannerBeanDefinitionParser解析器進行處理:

        public class ScannerNameSpaceHandler extends NamespaceHandlerSupport { @Override public void init() {  registerBeanDefinitionParser("scan", new CustomScannerBeanDefinitionParser()); }}

        3、自定義CustomScannerBeanDefinitionParser解析器:

        public class CustomScannerBeanDefinitionParser extends AbstractBeanDefinitionParser { @Override protected AbstractBeanDefinition parseInternal(Element element, ParserContext parserContext) {  BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(CustomScannerConfigurer.class);  ClassLoader classLoader = ClassUtils.getDefaultClassLoader();  try {   String annotationClassName = element.getAttribute("annotation");   if (StringUtils.hasText(annotationClassName)) {    Class annotationClass = (Class) classLoader      .loadClass(annotationClassName);    builder.addPropertyValue("annotationClass", annotationClass);   }  } catch (Exception ex) {   XmlReaderContext readerContext = parserContext.getReaderContext();   readerContext.error(ex.getMessage(), readerContext.extractSource(element), ex.getCause());  }  builder.addPropertyValue("basePackage", element.getAttribute("base-package"));  return builder.getBeanDefinition(); }}

        parseInternal()方法解析標簽,然后生成一個BeanDefinitionSpring會自動將其注冊到IoC容器中。如果標簽只會注冊單個Bean,這里是需要返回注冊Bean對應的BeanDefinition即可;如果是多個情況,這里一般是注冊一個配置類,將標簽配置的屬性注入到配置類中,然后由配置類統一處理。

        4、自定義CustomScannerConfigurer配置類:

        public class CustomScannerConfigurer  implements BeanDefinitionRegistryPostProcessor, InitializingBean { private String basePackage; private Class annotationClass; @Override public void afterPropertiesSet() throws Exception {  //參數校驗  notNull(this.basePackage, "Property "basePackage" is required"); } @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) { } @Override public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {  ClassPathBeanDefinitionScanner scanner =    new ClassPathBeanDefinitionScanner(registry, false);  scanner.addIncludeFilter(new AnnotationTypeFilter(annotationClass));  scanner.setIncludeAnnotationConfig(false);  int beanCount = scanner.scan(basePackage);  registry.getBeanDefinitionNames(); } public String getBasePackage() {  return basePackage; } public void setBasePackage(String basePackage) {  this.basePackage = basePackage; } public Class getAnnotationClass() {  return annotationClass; } public void setAnnotationClass(Class annotationClass) {  this.annotationClass = annotationClass; }}

        CustomScannerConfigurer實現了BeanDefinitionRegistryPostProcessor, InitializingBean兩個接口,之前分析過這兩個接口。重點在BeanDefinitionRegistryPostProcessor這個接口,其是一個BeanFactoryPostProcessor類型擴展,可以向IoC容器注冊BeanDefinition。在postProcessBeanDefinitionRegistry()方法中創建一個ClassPathBeanDefinitionScanner對象,并將標簽中配置設置進去,即可實現掃描指定包路徑下帶有指定注解的Bean

        5、xsd是標簽描述文件,NamespaceHandler則是標簽后臺處理邏輯入口,現在需要將兩者進行關聯,在resources/META-INF目錄下創建兩個文件:Spring.schemasSpring.handlers,分別指定xsd文件位置和NamespaceHandler位置,這樣就實現了標簽和后臺邏輯關聯,其內容見下:

        Spring.schemashttp\://www.simon.org/schema/scan.xsd=META-INF/custom-scan.xsd
        Spring.handlershttp\://www.simon.org/schema/scan=customschema.demo03.ScannerNameSpaceHandler

        自定義標簽描述以及對于的后臺處理邏輯都配置完成,下面我們就開始進行測試。

        1、首先,定義個注解,用于在掃描Bean時過濾使用:

        @Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documented@Indexedpublic @interface MyComponent { String value() default "";}

        2、在customschema.demo03.bean包路徑下定義三個類:TestService01TestService02TestService03,將后面兩個類使用@MyComponent注解標注下;

        3、編寫SpringXml配置文件,這里就可以使用我們剛才自定義的標簽:

         

        4、測試用例:

        @Testpublic void test01() { ApplicationContext context = new ClassPathXmlApplicationContext("custom-schema.xml"); Arrays.stream(context.getBeanDefinitionNames()).forEach(System.out::println);}

        從輸出結果就可以看到,TestService01由于沒有帶有@MyComponent注解,所以沒有注冊,TestService02TestService03都會被注冊到容器中。

        關鍵詞:

        首頁
        頻道
        底部
        頂部
        閱讀下一篇

        久久亚洲私人国产精品| 亚洲精品国产肉丝袜久久| 亚洲色欲久久久久综合网| 亚洲人色大成年网站在线观看| 国精无码欧精品亚洲一区| 亚洲中文字幕在线观看| 亚洲 自拍 另类小说综合图区| 亚洲精品9999久久久久无码 | 亚洲精品一区二区三区四区乱码| 亚洲AV区无码字幕中文色| 亚洲日韩精品一区二区三区| 亚洲国产精品尤物yw在线| 亚洲av综合av一区二区三区| 亚洲av乱码一区二区三区按摩 | 亚洲欧洲第一a在线观看| 亚洲天堂在线播放| 亚洲视频免费一区| 亚洲网红精品大秀在线观看| 亚洲性69影院在线观看| 亚洲一区二区三区在线| 中文字幕亚洲综合久久综合| 亚洲heyzo专区无码综合| 亚洲伊人久久大香线蕉影院| 亚洲熟妇av一区| 亚洲国产成a人v在线观看 | 亚洲国产精品综合久久网络| 亚洲av日韩av欧v在线天堂| 国产偷国产偷亚洲高清日韩| 亚洲日韩精品无码专区网址| 久久亚洲精品成人| 老汉色老汉首页a亚洲| 亚洲午夜在线一区| 亚洲乱码中文字幕在线| 亚洲AV网站在线观看| 亚洲永久精品ww47| 亚洲精品高清国产一线久久| 亚洲一区二区三区日本久久九| 亚洲国产女人aaa毛片在线| 亚洲日韩在线视频| 亚洲精品乱码久久久久蜜桃| 国产综合激情在线亚洲第一页|