在 spring boot 中,仅将 filter 声明为普通 bean 并不能使其参与 servlet 过滤链;必须通过 `filterregistrationbean` 显式注册,并可灵活设置 url 模式、优先级(order)和执行位置,从而实现内置与自定义 filter 的可控编排。
Spring Boot 默认会自动注册标注了 @Component 的 Filter 实现类,但其执行顺序由 Spring 容器内 Bean 的注册顺序或 @Order 注解决定,不可靠且无法精细干预(尤其当需插入到框架内置 Filter 之间时,如 CharacterEncodingFilter 和 HiddenHttpMethodFilter 之间)。你提到的 BeanDefinitionRegistryPostProcessor 方式虽能注册 Bean,但未将其绑定到 Servlet 容器——这正是 A/B/C Filter “已注册却未生效”的根本原因:它们只是 Spring Bean,而非 Servlet Filter。
✅ 正确做法是:为每个 Filter 创建对应的 FilterRegistrationBean
@Configuration
public class FilterConfig {
@Bean
public FilterRegistrationBean aFilterRegistration() {
FilterRegistrationBean registration = new FilterRegistrationBean<>();
registration.setFilter(new AFilter()); // 或注入 @Autowired AFilter aFilter
registration.addUrlPatterns("/*");
registration.setOrder(100); // 数值越小,优先级越高(早执行)
return registration;
}
@Bean
public FilterRegistrationBean bFilterRegistration() {
FilterRegistrationBean registration = new FilterRegistrationBean<>();
registration.setFilter(new BFilter());
registration.addUrlPatterns("/*");
registration.setOrder(200);
return registration;
}
@Bean
public FilterRegistrationBean cFilterRegistration() {
FilterRegistrationBean registration = new FilterRegistrationBean<>();
registration.setFilter(new CFilter());
registration.addUrlPatterns("/*");
registration.setOrder(300);
return registration;
}
// ✅ 用户自定义 DFilter,插入到 A 和 B 之间
@Bean
public FilterRegistrationBean dFilterRegistration() {
FilterRegistrationBean registration = new FilterRegistrationBean<>();
registration.setFilter(new DFilter());
registration.addUrlPatterns("/*");
registration.setOrder(150); // 100 < 150 < 200 → 执行顺序:A → D → B → C
return registration;
}
} ? 关键要点说明:
拟(如:A=100, B=200, C=300,则 D=150 即等效于 addFilterAfter(A).addFilterBefore(B)); 通过该方式,你不仅能精准控制 A/B/C/D 四个 Filter 的执行次序,还能与 Spring Security 的 SecurityFilterChain、Spring Boot 的 ErrorPageFilter 等内置 Filter 共存协同,真正实现可扩展、可维护、可测试的过滤器治理方案。