钓大鱼
123.36M · 2026-04-01
在Spring Boot项目里,Servlet、Filter、Listener是Java Web的核心组件,和传统Java Web项目比起来,它们的注册方式简单了不少~ Spring Boot提供了四种便捷注册方式,各自适配不同场景,下面给大家详细说说每种方式的实现、特点和注意点。
这是Spring Boot里最省事的注册方式:Spring Boot启动时,会自动扫描容器里所有属于Servlet、Filter、Listener的Spring Bean,直接把它们注册到嵌入式Servlet容器(比如Tomcat、Jetty)里。
实现示例:
// 1. 注册Servlet(作为Spring Bean)
@Component
public class MyServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().write("MyServlet Auto Registered");
}
}
// 2. 注册Filter(作为Spring Bean)
@Component
public class MyFilter implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
System.out.println("MyFilter Auto Filtering");
chain.doFilter(request, response);
}
}
// 3. 注册Listener(作为Spring Bean)
@Component
public class MyListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
System.out.println("ServletContext Initialized (Auto Registered)");
}
}
小问题:默认路径的局限及优化办法
/bean名称(就是类名首字母小写,比如上面的MyServlet,对应路径就是/myServlet);/*,意思是会拦截所有请求;优化小技巧:如果想改默认路径,要么结合方式二的RegistrationBean补充配置,要么直接用方式二、三、四,轻松实现自定义路径~
如果想对Servlet、Filter、Listener做更细致的控制——比如自定义映射路径、设置初始化参数、指定加载顺序,甚至排除某些不需要拦截的路径,方式一就无能为力了。好在Spring Boot提供的RegistrationBean系列类,能让我们实现完全自定义配置,想怎么配就怎么配。
先说下核心类:
拿Filter举个例子(其他两种组件用法类似):
@Configuration
public class WebConfig {
// 注册Filter,自定义路径和初始化参数
@Bean
public FilterRegistrationBean<MyFilter> myFilterRegistrationBean() {
FilterRegistrationBean<MyFilter> registrationBean = new FilterRegistrationBean<>();
registrationBean.setFilter(new MyFilter()); // 设置Filter实例
registrationBean.addUrlPatterns("/api/*"); // 自定义映射路径(仅拦截/api下的请求)
registrationBean.setInitParameter("filterName", "myFilter"); // 初始化参数
registrationBean.setOrder(1); // 设置加载顺序(值越小,加载越早)
return registrationBean;
}
// 注册Servlet,自定义路径和初始化参数
@Bean
public ServletRegistrationBean<MyServlet> myServletRegistrationBean() {
ServletRegistrationBean<MyServlet> registrationBean = new ServletRegistrationBean<>();
registrationBean.setServlet(new MyServlet());
registrationBean.addUrlMappings("/myServlet/custom"); // 自定义映射路径
registrationBean.setInitParameter("servletName", "myServlet");
registrationBean.setLoadOnStartup(1); // 启动时加载
return registrationBean;
}
// 注册Listener
@Bean
public ServletListenerRegistrationBean<MyListener> myListenerRegistrationBean() {
ServletListenerRegistrationBean<MyListener> registrationBean = new ServletListenerRegistrationBean<>();
registrationBean.setListener(new MyListener());
return registrationBean;
}
}
这是方式二(RegistrationBean手动注册)的注解替代方案,直接用Spring Boot提供的@ServletRegistration和@FilterRegistration两个注解,标注在Servlet和Filter类上,就能完成路径、初始化参数等自定义配置,适配喜欢注解式开发的场景。
核心说明:
实现示例:
// 用@ServletRegistration注册Servlet,替代ServletRegistrationBean
@ServletRegistration(urlPatterns = "/myServlet/annotation", loadOnStartup = 1)
@Component
public class MyServlet extends HttpServlet {
// 实现逻辑...
}
// 用@FilterRegistration注册Filter,替代FilterRegistrationBean
@FilterRegistration(urlPatterns = "/api/*", initParams = {@InitParam(name = "filterName", value = "myFilter")})
@Component
public class MyFilter implements Filter {
// 实现逻辑...
}
Servlet3.0规范本身就提供了三个注解——@WebServlet、@WebFilter、@WebListener,直接标注在组件类上,就能指定路径、名称这些配置。不过在Spring Boot项目里,得在启动类上加个@ServletComponentScan注解,开启对这三个注解的扫描,组件才能被自动注册上。
示例如下:
// 1. 启动类添加@ServletComponentScan,指定扫描包(可选,默认扫描当前包及子包)
@SpringBootApplication
@ServletComponentScan(basePackages = "com.example.web.component")
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
// 2. 用@WebServlet注册Servlet
@WebServlet(urlPatterns = "/webServlet", name = "WebServletDemo", loadOnStartup = 1)
public class WebServletDemo extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().write("WebServlet Annotated Registered");
}
}
// 3. 用@WebFilter注册Filter
@WebFilter(urlPatterns = "/web/*", filterName = "WebFilterDemo")
public class WebFilterDemo implements Filter {
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
System.out.println("WebFilter Annotated Filtering");
chain.doFilter(request, response);
}
}
// 4. 用@WebListener注册Listener
@WebListener
public class WebListenerDemo implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
System.out.println("ServletContext Initialized (WebListener Annotated)");
}
}
注意事项
| 注册方式 | 优点 | 局限 | 适用场景 |
|---|---|---|---|
| Spring Bean自动注册 | 配置最简单,不用写额外代码,省时间 | 路径等配置固定,不能自定义 | 快速开发、不需要自定义配置的简单场景 |
| RegistrationBean手动注册 | 配置超灵活,能完全控制组件的所有参数 | 需要写配置类,代码量稍微多一点 | 需要自定义路径、初始化参数、加载顺序的复杂场景 |
| @ServletRegistration和@FilterRegistration注解 | 注解式配置,无需写配置类,比RegistrationBean更简洁 | 仅支持Servlet和Filter,不支持Listener;需要spring-boot 3.5.x及以上版本 | 偏好注解开发、仅需注册Servlet和Filter的场景 |
| Servlet3.0注解+@ServletComponentScan | 注解式配置,代码简洁,符合Servlet规范 | 需要加扫描注解,Listener的配置比较有限 | 习惯用Servlet规范注解、需要简单自定义配置的场景 |