Spring-web源码解析之Initializer

简介: 基于4.1.7.RELEASESpring-WebApplicationInitializerAbstractContextLoaderInitializer / AbstractDi...

基于4.1.7.RELEASE

Spring-WebApplicationInitializer

AbstractContextLoaderInitializer / AbstractDispatcherServletInitializer / AbstractAnnotationConfigDispatcherServletInitializer

后3个类都是实现了WebApplicationInitializer接口,WebApplicationInitializer接口主要的作用是提供在Servlet 3.0+环境中对于ServletContext的可编程实现,可以跟web.xml混合使用,该接口的实现类会被SpringServletContainerInitializer自动调用,而SpringServletContainerInitializer则会被任意3.0+的Servlet自动启用,可以使用@Order注解来定义WebApplicationInitializer的执行顺序。

值得注意的一点是在tomcat的7.0.14(包含)以下版本时, tomcat会把url :“/”跟DefaultServlet绑定在一起且无法重写绑定关系。


需要注意的方法:

@Override
public void onStartup(ServletContext servletContext) throws ServletException {
   registerContextLoaderListener(servletContext);
}
在初始化Web Application的时候被调用,用于配置相关的ServletContext
在AbstractContextLoaderInitializer里,这里调用了registerContextLoaderListener方法,向其子类提供的WebApplicationContext注册了一个ContextLoaderListener
protected void registerContextLoaderListener(ServletContext servletContext) {
   WebApplicationContext rootAppContext = createRootApplicationContext();
   if (rootAppContext != null) {
      servletContext.addListener(new ContextLoaderListener(rootAppContext));
   }
}

而在AbstractDispatcherServletInitializer类中,onStartup则被重写,调用了本类的registerDispatcherServlet方法,注册了一个默认名字为dispatcher的DispatcherServlet,并且对Servlet进行了一些初始化工作。

protected void registerDispatcherServlet(ServletContext servletContext) {
   String servletName = getServletName();
   WebApplicationContext servletAppContext = createServletApplicationContext();

   DispatcherServlet dispatcherServlet = new DispatcherServlet(servletAppContext);
   ServletRegistration.Dynamic registration = servletContext.addServlet(servletName, dispatcherServlet);
   
   registration.setLoadOnStartup(1);
   registration.addMapping(getServletMappings());
   registration.setAsyncSupported(isAsyncSupported());

   Filter[] filters = getServletFilters();
   if (!ObjectUtils.isEmpty(filters)) {
      for (Filter filter : filters) {
         registerServletFilter(servletContext, filter);
      }
   }

   customizeRegistration(registration);
}

由代码中可以看出,在注册完dispatchServlet之后,设置启动优先级为1表明该Servlet随容器启动而初始化,添加Servlet的URL mapping即web.xml中的url-pattern选项,注册Servlet的filters并且调用了customizeRegistration方法来实现自定义配置ServletContext。其中

getServletName(), customizeRegistration() 可由实现类自己决定是否重写

createServletApplicationContext(),createRootApplicationContext()(继承自AbstractContextLoaderInitializer),getServletMappings()子类必须实现

有一个有意思的现象是在registerServletFilter中添加Filter时

int counter = -1;
while (counter == -1 || registration == null) {
   counter++;
   registration = servletContext.addFilter(filterName + "#" + counter, filter);
   Assert.isTrue(counter < 100,
         "Failed to register filter '" + filter + "'." +
         "Could the same Filter instance have been registered already?");
}

这表明,同一个名称的filter最多只能注册100个。

由于AbstractAnnotationConfigDispatcherServletInitializer继承了AbstractDispatcherServletInitializer,所以它满足了上面要求的createServletApplicationContext(),createRootApplicationContext()两个方法实现,而由于它是一个抽象类,getServletMappings()则被交给它的子类去实现。这两个方法的实现大同小异

Class<?>[] configClasses = getRootConfigClasses();
if (!ObjectUtils.isEmpty(configClasses)) {
   AnnotationConfigWebApplicationContext rootAppContext = new AnnotationConfigWebApplicationContext();
   rootAppContext.register(configClasses);
   return rootAppContext;
}else {
   return null;
}

如上所示,其不同在意获取configClasses的不同之处,一个是获取用于Servlet Application Context的配置,另一个是用于root Application Context的配置,无论是用在哪里,都是由其子类通过注解@Configuration和@Component来标识的。

目录
相关文章
|
23天前
|
负载均衡 监控 Java
Spring Cloud Gateway 全解析:路由配置、断言规则与过滤器实战指南
本文详细介绍了 Spring Cloud Gateway 的核心功能与实践配置。首先讲解了网关模块的创建流程,包括依赖引入(gateway、nacos 服务发现、负载均衡)、端口与服务发现配置,以及路由规则的设置(需注意路径前缀重复与优先级 order)。接着深入解析路由断言,涵盖 After、Before、Path 等 12 种内置断言的参数、作用及配置示例,并说明了自定义断言的实现方法。随后重点阐述过滤器机制,区分路由过滤器(如 AddRequestHeader、RewritePath、RequestRateLimiter 等)与全局过滤器的作用范围与配置方式,提
Spring Cloud Gateway 全解析:路由配置、断言规则与过滤器实战指南
|
9天前
|
缓存 安全 Java
Spring Security通用权限管理模型解析
Spring Security作为Spring生态的核心安全框架,结合RBAC与ACL权限模型,基于IoC与AOP构建灵活、可扩展的企业级权限控制体系,涵盖认证、授权流程及数据库设计、性能优化等实现策略。
58 0
|
9天前
|
缓存 安全 Java
Spring Security权限管理解析
Spring Security是Spring生态中的核心安全框架,采用认证与授权分离架构,提供高度可定制的权限管理方案。其基于过滤器链实现认证流程,通过SecurityContextHolder管理用户状态,并结合RBAC模型与动态权限决策,支持细粒度访问控制。通过扩展点如自定义投票器、注解式校验与前端标签,可灵活适配多租户、API网关等复杂场景。结合缓存优化与无状态设计,适用于高并发与前后端分离架构。
64 0
|
16天前
|
人工智能 Java 开发者
【Spring】原理解析:Spring Boot 自动配置
Spring Boot通过“约定优于配置”的设计理念,自动检测项目依赖并根据这些依赖自动装配相应的Bean,从而解放开发者从繁琐的配置工作中解脱出来,专注于业务逻辑实现。
|
23天前
|
SQL Java 数据库连接
Spring Data JPA 技术深度解析与应用指南
本文档全面介绍 Spring Data JPA 的核心概念、技术原理和实际应用。作为 Spring 生态系统中数据访问层的关键组件,Spring Data JPA 极大简化了 Java 持久层开发。本文将深入探讨其架构设计、核心接口、查询派生机制、事务管理以及与 Spring 框架的集成方式,并通过实际示例展示如何高效地使用这一技术。本文档约1500字,适合有一定 Spring 和 JPA 基础的开发者阅读。
115 0
|
9天前
|
Java 数据库 数据安全/隐私保护
Spring Boot四层架构深度解析
本文详解Spring Boot四层架构(Controller-Service-DAO-Database)的核心思想与实战应用,涵盖职责划分、代码结构、依赖注入、事务管理及常见问题解决方案,助力构建高内聚、低耦合的企业级应用。
195 0
|
21天前
|
Kubernetes Java 微服务
Spring Cloud 微服务架构技术解析与实践指南
本文档全面介绍 Spring Cloud 微服务架构的核心组件、设计理念和实现方案。作为构建分布式系统的综合工具箱,Spring Cloud 为微服务架构提供了服务发现、配置管理、负载均衡、熔断器等关键功能的标准化实现。本文将深入探讨其核心组件的工作原理、集成方式以及在实际项目中的最佳实践,帮助开发者构建高可用、可扩展的分布式系统。
206 0
|
22天前
|
安全 Java 数据安全/隐私保护
Spring Security 核心技术解析与实践指南
本文档深入探讨 Spring Security 框架的核心架构、关键组件和实际应用。作为 Spring 生态系统中负责安全认证与授权的关键组件,Spring Security 为 Java 应用程序提供了全面的安全服务。本文将系统介绍其认证机制、授权模型、过滤器链原理、OAuth2 集成以及最佳实践,帮助开发者构建安全可靠的企业级应用。
78 0
|
1月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。

推荐镜像

更多
  • DNS