SpringBoot中注册Servlet、Filter和Listener(代码和注解两种方式)

简介: SpringBoot中注册Servlet、Filter和Listener(代码和注解两种方式)

由于SpringBoot默认是以jar包的方式启动嵌入式的Servlet容器来启动SpringBoot的web应用,没有web.xml文件。

故而采用其他方式来注册三大组件:代码注册和注解注册。

【1】ServletRegistrationBean注册Servlet

查看ServletRegistrationBean源码如下:

/**
 * A {@link ServletContextInitializer} to register {@link Servlet}s in a Servlet 3.0+
 * container. Similar to the {@link ServletContext#addServlet(String, Servlet)
 * registration} features provided by {@link ServletContext} but with a Spring Bean
 * friendly design.
 * <p>
 * The {@link #setServlet(Servlet) servlet} must be specified before calling
 * {@link #onStartup}. URL mapping can be configured used {@link #setUrlMappings} or
 * omitted when mapping to '/*' (unless
 * {@link #ServletRegistrationBean(Servlet, boolean, String...) alwaysMapUrl} is set to
 * {@code false}). The servlet name will be deduced if not specified.
 *
 * @author Phillip Webb
 * @since 1.4.0
 * @see ServletContextInitializer
 * @see ServletContext#addServlet(String, Servlet)
 */
public class ServletRegistrationBean extends RegistrationBean {
  private static final Log logger = LogFactory.getLog(ServletRegistrationBean.class);
  private static final String[] DEFAULT_MAPPINGS = { "/*" };
  private Servlet servlet;
  private Set<String> urlMappings = new LinkedHashSet<String>();
  private boolean alwaysMapUrl = true;
  private int loadOnStartup = -1;
  private MultipartConfigElement multipartConfig;
  //...
}

注册示例如下:

 @Bean
 public ServletRegistrationBean myServlet(){
      ServletRegistrationBean registrationBean = new ServletRegistrationBean(new MyServlet(),"/myServlet");
      registrationBean.setLoadOnStartup(1);
      return registrationBean;
  }

【2】FilterRegistrationBean注册Filter

查看FilterRegistrationBean源码如下:

/**
 * A {@link ServletContextInitializer} to register {@link Filter}s in a Servlet 3.0+
 * container. Similar to the {@link ServletContext#addFilter(String, Filter) registration}
 * features provided by {@link ServletContext} but with a Spring Bean friendly design.
 * <p>
 * The {@link #setFilter(Filter) Filter} must be specified before calling
 * {@link #onStartup(ServletContext)}. Registrations can be associated with
 * {@link #setUrlPatterns URL patterns} and/or servlets (either by {@link #setServletNames
 * name} or via a {@link #setServletRegistrationBeans ServletRegistrationBean}s. When no
 * URL pattern or servlets are specified the filter will be associated to '/*'. The filter
 * name will be deduced if not specified.
 *
 * @author Phillip Webb
 * @since 1.4.0
 * @see ServletContextInitializer
 * @see ServletContext#addFilter(String, Filter)
 * @see DelegatingFilterProxyRegistrationBean
 */
public class FilterRegistrationBean extends AbstractFilterRegistrationBean {
  /**
   * Filters that wrap the servlet request should be ordered less than or equal to this.
   */
  public static final int REQUEST_WRAPPER_FILTER_MAX_ORDER = AbstractFilterRegistrationBean.REQUEST_WRAPPER_FILTER_MAX_ORDER;
  private Filter filter;
  //...
}

注册示例如下:

@Bean
public FilterRegistrationBean myFilter(){
     FilterRegistrationBean registrationBean = new FilterRegistrationBean();
     registrationBean.setFilter(new MyFilter());
     registrationBean.setUrlPatterns(Arrays.asList("/hello","/myServlet"));
     return registrationBean;
 }

【3】ServletListenerRegistrationBean注册Listener

查看ServletListenerRegistrationBean源码如下:

/**
 * A {@link ServletContextInitializer} to register {@link EventListener}s in a Servlet
 * 3.0+ container. Similar to the {@link ServletContext#addListener(EventListener)
 * registration} features provided by {@link ServletContext} but with a Spring Bean
 * friendly design.
 *
 * This bean can be used to register the following types of listener:
 * <ul>
 * <li>{@link ServletContextAttributeListener}</li>
 * <li>{@link ServletRequestListener}</li>
 * <li>{@link ServletRequestAttributeListener}</li>
 * <li>{@link HttpSessionAttributeListener}</li>
 * <li>{@link HttpSessionListener}</li>
 * <li>{@link ServletContextListener}</li>
 * </ul>
 *
 * @param <T> the type of listener
 * @author Dave Syer
 * @author Phillip Webb
 * @since 1.4.0
 */
public class ServletListenerRegistrationBean<T extends EventListener>
    extends RegistrationBean {
  private static final Log logger = LogFactory
      .getLog(ServletListenerRegistrationBean.class);
  private static final Set<Class<?>> SUPPORTED_TYPES;
  static {
    Set<Class<?>> types = new HashSet<Class<?>>();
    types.add(ServletContextAttributeListener.class);
    types.add(ServletRequestListener.class);
    types.add(ServletRequestAttributeListener.class);
    types.add(HttpSessionAttributeListener.class);
    types.add(HttpSessionListener.class);
    types.add(ServletContextListener.class);
    SUPPORTED_TYPES = Collections.unmodifiableSet(types);
  }
  private T listener;
  //...
}

注册Listener示例如下:

 @Bean
    public ServletListenerRegistrationBean myListener(){
        ServletListenerRegistrationBean<MyListener> registrationBean = new ServletListenerRegistrationBean<>(new MyListener());
        return registrationBean;
    }

【4】SpringMVC的DispatcherServlet默认注册机制

SpringBoot帮我们自动SpringMVC的时候,自动的注册SpringMVC的前端控制器:DIspatcherServlet。

查看DispatcherServletAutoConfiguration源码如下:

/**
 * {@link EnableAutoConfiguration Auto-configuration} for the Spring
 * {@link DispatcherServlet}. Should work for a standalone application where an embedded
 * servlet container is already present and also for a deployable application using
 * {@link SpringBootServletInitializer}.
 *
 * @author Phillip Webb
 * @author Dave Syer
 * @author Stephane Nicoll
 * @author Brian Clozel
 */
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE)
@Configuration
@ConditionalOnWebApplication
@ConditionalOnClass(DispatcherServlet.class)
@AutoConfigureAfter(EmbeddedServletContainerAutoConfiguration.class)
public class DispatcherServletAutoConfiguration {
  /*
   * The bean name for a DispatcherServlet that will be mapped to the root URL "/"
   */
  public static final String DEFAULT_DISPATCHER_SERVLET_BEAN_NAME = "dispatcherServlet";
  /*
   * The bean name for a ServletRegistrationBean for the DispatcherServlet "/"
   */
  public static final String DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME = "dispatcherServletRegistration";
  @Configuration
  @Conditional(DefaultDispatcherServletCondition.class)
  @ConditionalOnClass(ServletRegistration.class)
  @EnableConfigurationProperties(WebMvcProperties.class)
  protected static class DispatcherServletConfiguration {
    private final WebMvcProperties webMvcProperties;
    public DispatcherServletConfiguration(WebMvcProperties webMvcProperties) {
      this.webMvcProperties = webMvcProperties;
    }
    @Bean(name = DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)
    public DispatcherServlet dispatcherServlet() {
      DispatcherServlet dispatcherServlet = new DispatcherServlet();
      dispatcherServlet.setDispatchOptionsRequest(
          this.webMvcProperties.isDispatchOptionsRequest());
      dispatcherServlet.setDispatchTraceRequest(
          this.webMvcProperties.isDispatchTraceRequest());
      dispatcherServlet.setThrowExceptionIfNoHandlerFound(
          this.webMvcProperties.isThrowExceptionIfNoHandlerFound());
      return dispatcherServlet;
    }
    @Bean
    @ConditionalOnBean(MultipartResolver.class)
    @ConditionalOnMissingBean(name = DispatcherServlet.MULTIPART_RESOLVER_BEAN_NAME)
    public MultipartResolver multipartResolver(MultipartResolver resolver) {
      // Detect if the user has created a MultipartResolver but named it incorrectly
      return resolver;
    }
  }
  @Configuration
  @Conditional(DispatcherServletRegistrationCondition.class)
  @ConditionalOnClass(ServletRegistration.class)
  @EnableConfigurationProperties(WebMvcProperties.class)
  @Import(DispatcherServletConfiguration.class)
  protected static class DispatcherServletRegistrationConfiguration {
    private final ServerProperties serverProperties;
    private final WebMvcProperties webMvcProperties;
    private final MultipartConfigElement multipartConfig;
    public DispatcherServletRegistrationConfiguration(
        ServerProperties serverProperties, WebMvcProperties webMvcProperties,
        ObjectProvider<MultipartConfigElement> multipartConfigProvider) {
      this.serverProperties = serverProperties;
      this.webMvcProperties = webMvcProperties;
      this.multipartConfig = multipartConfigProvider.getIfAvailable();
    }
    // 这里注册DispatcherServlet
    @Bean(name = DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME)
    @ConditionalOnBean(value = DispatcherServlet.class, name = DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)
    public ServletRegistrationBean dispatcherServletRegistration(
        DispatcherServlet dispatcherServlet) {
      ServletRegistrationBean registration = new ServletRegistrationBean(
          dispatcherServlet, this.serverProperties.getServletMapping());
      registration.setName(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME);
      registration.setLoadOnStartup(
          this.webMvcProperties.getServlet().getLoadOnStartup());
      if (this.multipartConfig != null) {
        registration.setMultipartConfig(this.multipartConfig);
      }
      return registration;
    }
  }
  @Order(Ordered.LOWEST_PRECEDENCE - 10)
  private static class DefaultDispatcherServletCondition extends SpringBootCondition {
    @Override
    public ConditionOutcome getMatchOutcome(ConditionContext context,
        AnnotatedTypeMetadata metadata) {
      ConditionMessage.Builder message = ConditionMessage
          .forCondition("Default DispatcherServlet");
      ConfigurableListableBeanFactory beanFactory = context.getBeanFactory();
      List<String> dispatchServletBeans = Arrays.asList(beanFactory
          .getBeanNamesForType(DispatcherServlet.class, false, false));
      if (dispatchServletBeans.contains(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)) {
        return ConditionOutcome.noMatch(message.found("dispatcher servlet bean")
            .items(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME));
      }
      if (beanFactory.containsBean(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)) {
        return ConditionOutcome
            .noMatch(message.found("non dispatcher servlet bean")
                .items(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME));
      }
      if (dispatchServletBeans.isEmpty()) {
        return ConditionOutcome
            .match(message.didNotFind("dispatcher servlet beans").atAll());
      }
      return ConditionOutcome.match(message
          .found("dispatcher servlet bean", "dispatcher servlet beans")
          .items(Style.QUOTE, dispatchServletBeans)
          .append("and none is named " + DEFAULT_DISPATCHER_SERVLET_BEAN_NAME));
    }
  }
  @Order(Ordered.LOWEST_PRECEDENCE - 10)
  private static class DispatcherServletRegistrationCondition
      extends SpringBootCondition {
    @Override
    public ConditionOutcome getMatchOutcome(ConditionContext context,
        AnnotatedTypeMetadata metadata) {
      ConfigurableListableBeanFactory beanFactory = context.getBeanFactory();
      ConditionOutcome outcome = checkDefaultDispatcherName(beanFactory);
      if (!outcome.isMatch()) {
        return outcome;
      }
      return checkServletRegistration(beanFactory);
    }
    private ConditionOutcome checkDefaultDispatcherName(
        ConfigurableListableBeanFactory beanFactory) {
      List<String> servlets = Arrays.asList(beanFactory
          .getBeanNamesForType(DispatcherServlet.class, false, false));
      boolean containsDispatcherBean = beanFactory
          .containsBean(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME);
      if (containsDispatcherBean
          && !servlets.contains(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME)) {
        return ConditionOutcome
            .noMatch(startMessage().found("non dispatcher servlet")
                .items(DEFAULT_DISPATCHER_SERVLET_BEAN_NAME));
      }
      return ConditionOutcome.match();
    }
    private ConditionOutcome checkServletRegistration(
        ConfigurableListableBeanFactory beanFactory) {
      ConditionMessage.Builder message = startMessage();
      List<String> registrations = Arrays.asList(beanFactory
          .getBeanNamesForType(ServletRegistrationBean.class, false, false));
      boolean containsDispatcherRegistrationBean = beanFactory
          .containsBean(DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME);
      if (registrations.isEmpty()) {
        if (containsDispatcherRegistrationBean) {
          return ConditionOutcome
              .noMatch(message.found("non servlet registration bean").items(
                  DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME));
        }
        return ConditionOutcome
            .match(message.didNotFind("servlet registration bean").atAll());
      }
      if (registrations
          .contains(DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME)) {
        return ConditionOutcome.noMatch(message.found("servlet registration bean")
            .items(DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME));
      }
      if (containsDispatcherRegistrationBean) {
        return ConditionOutcome
            .noMatch(message.found("non servlet registration bean").items(
                DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME));
      }
      return ConditionOutcome.match(message.found("servlet registration beans")
          .items(Style.QUOTE, registrations).append("and none is named "
              + DEFAULT_DISPATCHER_SERVLET_REGISTRATION_BEAN_NAME));
    }
    private ConditionMessage.Builder startMessage() {
      return ConditionMessage.forCondition("DispatcherServlet Registration");
    }
  }
}

【5】Servlet3.0下注册三大组件的方式

在上面的源码注释中,频繁见到如下字样:

 A {@link ServletContextInitializer} to register {@link Filter}s in a Servlet 3.0+
 * container. Similar to the {@link ServletContext#addFilter(String, Filter) registration}
 * features provided by {@link ServletContext} but with a Spring Bean friendly design.

即,SpringBoot提供的三大组件添加方式只是一种更为友好的设计方式,和Servlet3.0中使用ServletContext#addFilter方式一样!

【6】SpringBoot下使用注册注册三大组件

该种方式与【5】中Servlet、Filter和Listener(注解注册)是一致的,只是需要在主配置类上添加@ServletComponentScan注册。

示例如下:

@SpringBootApplication
@ServletComponentScan
public class SpringBootApplication {
  public static void main(String[] args) {
    SpringApplication.run(SpringBootApplication.class, args);
  }
  @Bean
  public ViewResolver myViewReolver(){
    return new MyViewResolver();
  }
  public static class MyViewResolver implements ViewResolver{
    @Override
    public View resolveViewName(String viewName, Locale locale) throws Exception {
      return null;
    }
  }
}

然后就可以在Servlet、Filter和Listener上使用注解进行注册!


目录
相关文章
|
5月前
|
安全 Java 应用服务中间件
Spring Boot + Java 21:内存减少 60%,启动速度提高 30% — 零代码
通过调整三个JVM和Spring Boot配置开关,无需重写代码即可显著优化Java应用性能:内存减少60%,启动速度提升30%。适用于所有在JVM上运行API的生产团队,低成本实现高效能。
655 3
|
8月前
|
监控 Java 数据安全/隐私保护
阿里面试:SpringBoot启动时, 如何执行扩展代码?你们项目 SpringBoot 进行过 哪些 扩展?
阿里面试:SpringBoot启动时, 如何执行扩展代码?你们项目 SpringBoot 进行过 哪些 扩展?
|
4月前
|
缓存 安全 Java
《深入理解Spring》过滤器(Filter)——Web请求的第一道防线
Servlet过滤器是Java Web核心组件,可在请求进入容器时进行预处理与响应后处理,适用于日志、认证、安全、跨域等全局性功能,具有比Spring拦截器更早的执行时机和更广的覆盖范围。
|
8月前
|
Java 数据库连接 数据库
Spring boot 使用mybatis generator 自动生成代码插件
本文介绍了在Spring Boot项目中使用MyBatis Generator插件自动生成代码的详细步骤。首先创建一个新的Spring Boot项目,接着引入MyBatis Generator插件并配置`pom.xml`文件。然后删除默认的`application.properties`文件,创建`application.yml`进行相关配置,如设置Mapper路径和实体类包名。重点在于配置`generatorConfig.xml`文件,包括数据库驱动、连接信息、生成模型、映射文件及DAO的包名和位置。最后通过IDE配置运行插件生成代码,并在主类添加`@MapperScan`注解完成整合
1404 1
Spring boot 使用mybatis generator 自动生成代码插件
|
8月前
|
Java 调度 流计算
基于Java 17 + Spring Boot 3.2 + Flink 1.18的智慧实验室管理系统核心代码
这是一套基于Java 17、Spring Boot 3.2和Flink 1.18开发的智慧实验室管理系统核心代码。系统涵盖多协议设备接入(支持OPC UA、MQTT等12种工业协议)、实时异常检测(Flink流处理引擎实现设备状态监控)、强化学习调度(Q-Learning算法优化资源分配)、三维可视化(JavaFX与WebGL渲染实验室空间)、微服务架构(Spring Cloud构建分布式体系)及数据湖建设(Spark构建实验室数据仓库)。实际应用中,该系统显著提升了设备调度效率(响应时间从46分钟降至9秒)、设备利用率(从41%提升至89%),并大幅减少实验准备时间和维护成本。
439 0
|
12月前
|
Cloud Native Java Nacos
springcloud/springboot集成NACOS 做注册和配置中心以及nacos源码分析
通过本文,我们详细介绍了如何在 Spring Cloud 和 Spring Boot 中集成 Nacos 进行服务注册和配置管理,并对 Nacos 的源码进行了初步分析。Nacos 作为一个强大的服务注册和配置管理平台,为微服务架构提供
4650 14
|
XML 前端开发 Java
SpringBoot整合Flowable【04】- 通过代码控制流程流转
本文介绍了如何使用Flowable的Java API控制流程流转,基于前文构建的绩效流程模型。首先,通过Flowable-UI导出模型文件并部署到Spring Boot项目中。接着,详细讲解了如何通过代码部署、启动和审批流程,涉及`RepositoryService`、`RuntimeService`和`TaskService`等核心服务类的使用。最后,通过实际操作演示了流程从部署到完成的全过程,并简要说明了相关数据库表的变化。本文帮助读者初步掌握Flowable在实际业务中的应用,后续将深入探讨更多高级功能。
1940 0
SpringBoot整合Flowable【04】-  通过代码控制流程流转
|
Java API Spring
在 Spring 配置文件中配置 Filter 的步骤
【10月更文挑战第21天】在 Spring 配置文件中配置 Filter 是实现请求过滤的重要手段。通过合理的配置,可以灵活地对请求进行处理,满足各种应用需求。还可以根据具体的项目要求和实际情况,进一步深入研究和优化 Filter 的配置,以提高应用的性能和安全性。
|
搜索推荐 Java Spring
Spring Filter深度解析
【10月更文挑战第21天】Spring Filter 是 Spring 框架中非常重要的一部分,它为请求处理提供了灵活的控制和扩展机制。通过合理配置和使用 Filter,可以实现各种个性化的功能,提升应用的安全性、可靠性和性能。还可以结合具体的代码示例和实际应用案例,进一步深入探讨 Spring Filter 的具体应用和优化技巧,使对它的理解更加全面和深入。
|
前端开发 Java 数据库
玩转springboot之springboot注册servlet
在Spring Boot中注册Servlet非常灵活,可以通过 `@WebServlet`注解快速注册,也可以通过 `ServletRegistrationBean`进行细粒度控制。通过这两种方式,可以满足各种场景下的需求,确保应用能够高效处理HTTP请求。
1225 14