
1. 前言
我們在上一篇對Mybatis如何將Mapper介面注入Spring IoC進行了分析,有同學問胖哥這個有什么用,這個作用其實挺大的,比如讓你實作一個類似@Controller的注解(或者繼承某個統一介面)來完成比如定時任務的統一注入或者Websocket處理器的統一注入等這種將某種共性的Bean動態注入,
// 模仿 Controller
@XBean(description = "ETL JOB")
public class JobShedule {
@Caller(cron = "* * 0/5 * * ?")
public void exec(){
// job
}
}
以上偽代碼就是一個模仿Controller的定時任務Bean,
2. 設計思路
詳細的開發設計思路我已經總結好了,各位同學只要按部就班就可以實作這個功能了,
2.1 定義掃描注解
定義一個類似@MappScan的進行匯入自定義ImportBeanDefinitionRegistrar,并指定掃描包范圍,
@Documented
@Inherited
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
@Import(XBeanDefinitionRegistrar.class)
public @interface XBeanScan {
String[] basePackages();
}
我們自定義了一個掃描注解@XBeanScan,它有兩個作用:
- 通過
basePackages指定掃描包的范圍, - 匯入我們自定義
ImportBeanDefinitionRegistrar的實作XBeanDefinitionRegistrar,
2.2 定義目標Bean的通用標記
通常我們可以選擇一個標識介面,所有其實作類都會注入Spring IoC;或者用更加方便的注解,所有被該注解標記的類都將注入Spring IoC,這里我們使用更加靈活方便的注解,實作了一個@XBean標記注解:
@Documented
@Inherited
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
public @interface XBean {
String description() default "";
}
2.3 實作掃描器
Spring框架為我們提供了掃描器來注冊被標記的Bean,它就是上節提到的ClassPathBeanDefinitionScanner,我們繼承它進行稍加改造:
public class XBeanDefinitionScanner extends ClassPathBeanDefinitionScanner {
public XBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters) {
super(registry, useDefaultFilters);
super.addIncludeFilter(new AnnotationTypeFilter(XBean.class));
}
}
這里我們不使用默認的過濾器,我們指定了掃描器掃描的目標為被@XBean標記的那些Bean,
2.4 實作 Bean 注冊機
重頭戲來了,我們需要將2.1到2.3定義的這些組件在ImportBeanDefinitionRegistrar的實作中組裝起來,
/**
* The type X bean definition registrar.
*
* @author felord.cn
* @since 2020 /9/18 22:59
*/
public class XBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar, ResourceLoaderAware {
private ResourceLoader resourceLoader;
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
// 不使用默認過濾器
XBeanDefinitionScanner xBeanDefinitionScanner = new XBeanDefinitionScanner(registry, false);
xBeanDefinitionScanner.setResourceLoader(resourceLoader);
// 掃描XBeanScan注解指定的包
xBeanDefinitionScanner.scan(getBasePackagesToScan(importingClassMetadata));
}
@Override
public void setResourceLoader(ResourceLoader resourceLoader) {
this.resourceLoader = resourceLoader;
}
/**
* 獲取{@link XBeanScan}中宣告的掃描包路徑
* @param metadata the meta
* @return 包路徑陣列
*/
private String[] getBasePackagesToScan(AnnotationMetadata metadata) {
String name = XBeanScan.class.getName();
AnnotationAttributes attributes = AnnotationAttributes.fromMap(metadata.getAnnotationAttributes(name, true));
Assert.notNull(attributes, () -> "No auto-configuration attributes found. Is " + metadata.getClassName()
+ " annotated with " + ClassUtils.getShortName(name) + "?");
return attributes.getStringArray("basePackages");
}
}
從注解元資料importingClassMetadata決議我們需要的掃描路徑basePackages等元資料,然后讓掃描器在該路徑掃描即可,
2.5 使用
在具有@Configuration標記的類或者Spring Boot的Main類上使用@XBeanScan即可,是不是非常簡單!
其實
@ComponentScan提供類似的功能,
3. 總結
本篇是對上一篇理論的具體應用,說實話上一篇比較枯燥甚至抓不住重點,但是有時候理論就是這樣的,一旦你結合本篇來看你會恍然大悟,如果你需要更加細粒度控制就加上那些BeanDefinitionRegistryPostProcessor和FactoryBean等Spring提供的功能性介面,從這兩篇中更多需要你學習的是如何從閱讀原始碼中觸類旁通,來利用已有的組件來實作自己的邏輯,這對你的提高是極大的,好了今天就到這里,多多關注:碼農小胖哥 更多干貨等著你,
關注公眾號:Felordcn 獲取更多資訊
個人博客:https://felord.cn
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/159812.html
標籤:Java
上一篇:計算機專業,為何在大一期間學習的是C語言,而不學習其他語言?
下一篇:Java學習day09
