在
SpringBoot
系列文章的《第七章:過濾器、監聽器、攔截器》中,小技巧中指出,可以使用@Order
設置過濾器的執行順序。因爲沒有本身求證過,看了相關材料後,想固然的寫進了文章中,這個進行更正下。java
經過過濾器名稱
和設置@Order
的方法都是不行的。抱歉了,各位。以後在編寫文章時,會本着負責且持着大膽猜想當心求證的態度,會對相關事項進行覈對的!再次,抱歉,誤導了你們web
這裏要感謝簡書網友:形而上學本尊,指出此錯誤!再次感謝!spring
《第七章:過濾器、監聽器、攔截器》也有指出,利用
FilterRegistrationBean
能夠設置排序順序。那是否還有其餘方式呢。有的,只是這種方案不是很優雅。這裏簡單說明下。springboot
先說結論:能夠經過過濾器
的類名進行約定排序。微信
既然遇到了,那就簡單分析下使用
@WebFilter
和@ServletComponentScan
的啓動方式吧。app
首先咱們來看下,註解@ServletComponentScan
(刪除了相關注解):ide
@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
進行註冊。看見這些是否是很熟悉了。post
//部分代碼 static { List<ServletComponentHandler> servletComponentHandlers = new ArrayList<ServletComponentHandler>(); servletComponentHandlers.add(new WebServletHandler()); servletComponentHandlers.add(new WebFilterHandler()); servletComponentHandlers.add(new WebListenerHandler()); HANDLERS = Collections.unmodifiableList(servletComponentHandlers); }
關鍵看這個方法scanPackage
:this
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
系列教程,若有雷同,請多多包涵了。原創不易,碼字不易,還但願你們多多支持。若文中有所錯誤之處,還望提出,謝謝。
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/