前言
在
SpringBoot
系列文章的《第七章:過濾器、監聽器、攔截器》中,小技巧中指出,可使用@Order
設置過濾器的執行順序。由於沒有自己求證過,看了相關材料后,想當然的寫進了文章中,這個進行更正下。
通過過濾器名稱
和設置@Order
的方法都是不行的。抱歉了,各位。之后在編寫文章時,會本着負責且持着大膽猜測小心求證的態度,會對相關事項進行核對的!再次,抱歉,誤導了大家
這里要感謝簡書網友:形而上學本尊,指出此錯誤!再次感謝!
正確設置排序方式
《第七章:過濾器、監聽器、攔截器》也有指出,利用
FilterRegistrationBean
可以設置排序順序。那是否還有其他方式呢。有的,只是這種方案不是很優雅。這里簡單說明下。
先說結論:可以通過過濾器
的類名進行約定排序。
淺談ServletComponentScan注解的啟動方式
既然遇到了,那就簡單分析下使用
@WebFilter
和@ServletComponentScan
的啟動方式吧。
首先我們來看下,注解@ServletComponentScan
(刪除了相關注解):
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(ServletComponentScanRegistrar.class)
public @interface ServletComponentScan {
@AliasFor("basePackages")
String[] value() default {};
@AliasFor("value")
String[] basePackages() default {};
Class<?>[] basePackageClasses() default {};
}
簡單來說,此注解就是指定掃描路徑的,通過value
、basePackages
或者basePackageClasses
。主要還是看下ServletComponentScanRegistrar
類,這才是關鍵。
class ServletComponentScanRegistrar implements ImportBeanDefinitionRegistrar {
private static final String BEAN_NAME = "servletComponentRegisteringPostProcessor";
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
// 獲取包路徑
Set<String> packagesToScan = getPackagesToScan(importingClassMetadata);
// 若已注冊,則更新,否則新增
if (registry.containsBeanDefinition(BEAN_NAME)) {
updatePostProcessor(registry, packagesToScan);
} else {
addPostProcessor(registry, packagesToScan);
}
}
private void updatePostProcessor(BeanDefinitionRegistry registry, Set<String> packagesToScan) {
BeanDefinition definition = registry.getBeanDefinition(BEAN_NAME);
ValueHolder constructorArguments = definition.getConstructorArgumentValues().getGenericArgumentValue(Set.class);
@SuppressWarnings("unchecked")
Set<String> mergedPackages = (Set<String>) constructorArguments.getValue();
mergedPackages.addAll(packagesToScan);
constructorArguments.setValue(mergedPackages);
}
private void addPostProcessor(BeanDefinitionRegistry registry, Set<String> packagesToScan) {
GenericBeanDefinition beanDefinition = new GenericBeanDefinition();
// 設置類
beanDefinition.setBeanClass(ServletComponentRegisteringPostProcessor.class);
// 設置構造函數參數
beanDefinition.getConstructorArgumentValues().addGenericArgumentValue(packagesToScan);
beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
// 注冊
registry.registerBeanDefinition(BEAN_NAME, beanDefinition);
}
private Set<String> getPackagesToScan(AnnotationMetadata metadata) {
// 獲取注解ServletComponentScan的屬性信息
AnnotationAttributes attributes = AnnotationAttributes
.fromMap(metadata.getAnnotationAttributes(ServletComponentScan.class.getName()));
// 獲取屬性basePackages和basePackageClasses
String[] basePackages = attributes.getStringArray("basePackages");
Class<?>[] basePackageClasses = attributes.getClassArray("basePackageClasses");
Set<String> packagesToScan = new LinkedHashSet<String>();
packagesToScan.addAll(Arrays.asList(basePackages));
// basePackageClasses 最后也是根據basePackageClasses來獲取塔對應的包路徑
for (Class<?> basePackageClass : basePackageClasses) {
packagesToScan.add(ClassUtils.getPackageName(basePackageClass));
}
// 默認不填寫時,獲取的是被注解類所在包路徑,所以一般放在啟動類上
if (packagesToScan.isEmpty()) {
packagesToScan.add(ClassUtils.getPackageName(metadata.getClassName()));
}
return packagesToScan;
}
}
可以看見,它是一個ImportBeanDefinitionRegistrar
的實現類,ImportBeanDefinitionRegistrar
可以動態地裝載Bean
。再來看看ServletComponentRegisteringPostProcessor
類,此類是個BeanFactoryPostProcessor
,BeanFactory的后置處理器,簡單理解就是擴展點吧。啟動的時候會調用postProcessBeanFactory
方法。
ServletComponentRegisteringPostProcessor
源碼就不貼了,簡單來說,它的作用就是:掃描被@WebServlet
、@WebFilter
及@WebListener
的類,最后通過對應的ServletRegistrationBean
、FilterRegistrationBean
及ServletListenerRegistrationBean
進行注冊。看見這些是不是很熟悉了。
//部分代碼
static {
List<ServletComponentHandler> servletComponentHandlers = new ArrayList<ServletComponentHandler>();
servletComponentHandlers.add(new WebServletHandler());
servletComponentHandlers.add(new WebFilterHandler());
servletComponentHandlers.add(new WebListenerHandler());
HANDLERS = Collections.unmodifiableList(servletComponentHandlers);
}
關鍵看這個方法scanPackage
:
private void scanPackage(
ClassPathScanningCandidateComponentProvider componentProvider,
String packageToScan) {
for (BeanDefinition candidate : componentProvider
.findCandidateComponents(packageToScan)) {
if (candidate instanceof ScannedGenericBeanDefinition) {
for (ServletComponentHandler handler : HANDLERS) {
handler.handle(((ScannedGenericBeanDefinition) candidate),
(BeanDefinitionRegistry) this.applicationContext);
}
}
}
}
可以看見,通過componentProvider.findCandidateComponents(packageToScan)
方法獲取到對應的注解類,同時判斷是否為以上說的三種,最后調用其doHandle
方法完成注冊功能。以下是WebFilterHandler
的doHandler
方法。
現在,我們看看findCandidateComponents
方法怎么獲取對應注解類的。
斷點之后,可以看見是AnnotationConfigEmbeddedWebApplicationContext
類,
繼續斷點進去,最后是使用PathMatchingResourcePatternResolver
類進行資源獲取的。
通過遞歸的方式,獲取所有的類:
最后關鍵就是這個Arrays.sort(dirContents)
了。所以簡單來說,可以通過class類名來達到排序效果。但這種方案要限制類名,還是使用FilterRegistrationBean
之類的來設置吧。
總結
寫的可能有點亂也有點水,⊙﹏⊙‖∣。主要還是想糾正下原先的錯誤,O__O…。知其然知其所以然,還有很長的路要走。沒有寫里面的細節,只是大致講解了下。有興趣的可以自行跟蹤看看。
最后
目前互聯網上很多大佬都有
SpringBoot
系列教程,如有雷同,請多多包涵了。原創不易,碼字不易,還希望大家多多支持。若文中有所錯誤之處,還望提出,謝謝。
老生常談
- 個人QQ:
499452441
- 微信公眾號:
lqdevOps
個人博客:http://blog.lqdev.cn
原文地址:http://blog.lqdev.cn/2018/08/26/%E6%97%A5%E5%B8%B8%E7%A7%AF%E7%B4%AF/correct-webfilter/