Tomcat源码分析(四):ServletContext应用启动之核心组件初始化

本文主要是介绍Tomcat源码分析(四):ServletContext应用启动之核心组件初始化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

  • 由我的上篇文章Tomcat源码分析(三):ServletContext应用启动之配置解析可知,在Tomcat启动应用StandardContext时,是先通过listener事件机制的方式,交给ContextConfig,解析web.xml类来获取应用的配置信息,包含过滤器filter,监听器listener,如spring的ContextLoaderListener,servlet,如spring的DispatchServlet,以及session配置等。但是配置解析这边只是创建一个包装类,如过滤器的FilterDef,servlet的StandardWrapper,获取这些组件对应的类的全限定名以及init params等数据保存到包装类中,而不进行实际类对象的创建。

  • 其次在Servlet3.0之后,提供了WebApplicationInitializer接口来支持以编程方式配置web.xml的内容,故在ContextConfig中也会查找应用中的WebApplicationInitializer的实例类,然后保存到StandardContext的initializers中。注意在ContextConfig并不会对WebApplicationInitializer进行解析,即调用其onStartup方法,而是留在StardardContext中执行。

    /*** The ordered set of ServletContainerInitializers for this web application.*/// map的value为WebApplicationInitializer的集合
    private Map<ServletContainerInitializer,Set<Class<?>>> initializers =new LinkedHashMap<>();
    
  • 具体以上对象实例的创建,如servlet, filter,listener等,是之后在StandardContext中继续完成。即在StandardContext的startInternal方法中,以如下顺序继续完成启动和相关类对象实例的创建。

1. context初始化参数加载

  • context初始化参数主要包括两种,一种是在web.xml(或者WebApplicationInitializer,以下提到web.xml的类似)配置context-param设置,如下:

    <context-param>  <param-name>contextConfigLocation</param-name>  <param-value>classpath:spring/applicationContext.xml</param-value>  
    </context-param>
    
  • 另外一种则是在Server.xml的Context节点下面的Parameter节点配置,或者在Context.xml中配置,这些是提供某个param默认值的方式,可以被web.xml中的覆盖。然后在Digester解析规则会添加Parameter解析器,具体为在ContextRuleSet中定义,如下:

    digester.addObjectCreate(prefix + "Context/Parameter","org.apache.tomcat.util.descriptor.web.ApplicationParameter");
    digester.addSetProperties(prefix + "Context/Parameter");
    digester.addSetNext(prefix + "Context/Parameter","addApplicationParameter","org.apache.tomcat.util.descriptor.web.ApplicationParameter");
    
  • 在ContextConfig中已经解析好了在web.xml中配置的param,然后接着在StandardContext中,将以上两种方式获取的param填充到servletContext中:

    /*** Merge the context initialization parameters specified in the application* deployment descriptor with the application parameters described in the* server configuration, respecting the <code>override</code> property of* the application parameters appropriately.*/
    private void mergeParameters() {Map<String,String> mergedParams = new HashMap<>();// web.xmlString names[] = findParameters();for (int i = 0; i < names.length; i++) {mergedParams.put(names[i], findParameter(names[i]));}// Parameter节点ApplicationParameter params[] = findApplicationParameters();for (int i = 0; i < params.length; i++) {if (params[i].getOverride()) {if (mergedParams.get(params[i].getName()) == null) {mergedParams.put(params[i].getName(),params[i].getValue());}} else {mergedParams.put(params[i].getName(), params[i].getValue());}}// 填充到ServletContext中ServletContext sc = getServletContext();for (Map.Entry<String,String> entry : mergedParams.entrySet()) {sc.setInitParameter(entry.getKey(), entry.getValue());}}
    

2. SCI(ServletContainerInitializers)启动onStartup

  • 在ContextConfig查找到应用中的WebApplicationInitializer实现类中,填充保存在StandardContext的initializers集合中,然后在这步调用WebApplicationInitializer的onStartup方法,获取编程方式的应用配置信息。
    // Call ServletContainerInitializers
    for (Map.Entry<ServletContainerInitializer, Set<Class<?>>> entry :initializers.entrySet()) {try {// 在spring中,则是遍历WebApplicationInitializer集合,// 即entry.getValue(),// 调用WebApplicationInitializer的onStartup方法;// getServletContext()为传递当前应用的servletContextentry.getKey().onStartup(entry.getValue(),getServletContext());} catch (ServletException e) {log.error(sm.getString("standardContext.sciFail"), e);ok = false;break;}
    }
    

3. Listener监听器配置与调用

  • 这些一般是应用在web.xml和WebApplicationInitializer中配置的监听器,而不是Tomcat自身添加的监听器,如ContextConfig是Tomcat自身使用Digester解析server.xml时添加的监听器,用于解析配置文件。而这里说的监听器是应用在web.xml或者WebApplicationInitializer中,即配置文件中添加的,类型包括ServletContextListener,HttpSessionListener,ServletRequestListener等,如spring的ContextLoaderListener就是一个ServletContextListener接口的实现类。

  • 实现在StandardContext的listenerStart方法,核心实现如下:即包括:

    1. EventListener和LifeCycleListener的创建
    2. 生命周期监听器LifeCycleListener的调用
    /*** Configure the set of instantiated application event listeners* for this Context.* @return <code>true</code> if all listeners wre* initialized successfully, or <code>false</code> otherwise.*/
    public boolean listenerStart() {...// 1. 创建listener对象实例// Instantiate the required listenersString listeners[] = findApplicationListeners();Object results[] = new Object[listeners.length];boolean ok = true;for (int i = 0; i < results.length; i++) {...String listener = listeners[i];results[i] = getInstanceManager().newInstance(listener);...}...// 2. 分类:分为EventListener和LifeCycleListener,// 并保存到StandardContext中// EventListener是在应用运行过程中调用的// LifetCycleListener是应用启动,关闭等中调用的// Sort listeners in two arraysArrayList<Object> eventListeners = new ArrayList<>();ArrayList<Object> lifecycleListeners = new ArrayList<>();for (int i = 0; i < results.length; i++) {if ((results[i] instanceof ServletContextAttributeListener)|| (results[i] instanceof ServletRequestAttributeListener)|| (results[i] instanceof ServletRequestListener)|| (results[i] instanceof HttpSessionIdListener)|| (results[i] instanceof HttpSessionAttributeListener)) {eventListeners.add(results[i]);}if ((results[i] instanceof ServletContextListener)|| (results[i] instanceof HttpSessionListener)) {lifecycleListeners.add(results[i]);}}// Listener instances may have been added directly to this Context by// ServletContextInitializers and other code via the pluggability APIs.// Put them these listeners after the ones defined in web.xml and/or// annotations then overwrite the list of instances with the new, full// list.for (Object eventListener: getApplicationEventListeners()) {eventListeners.add(eventListener);}setApplicationEventListeners(eventListeners.toArray());for (Object lifecycleListener: getApplicationLifecycleListeners()) {lifecycleListeners.add(lifecycleListener);if (lifecycleListener instanceof ServletContextListener) {noPluggabilityListeners.add(lifecycleListener);}}setApplicationLifecycleListeners(lifecycleListeners.toArray());...// 3. 调用LifetCycleListener,处理应用启动初始化事件// 这里只调用LifetCycleListener,// 不调用EventListener// Send application start events// Ensure context is not nullgetServletContext();context.setNewServletContextListenerAllowed(false);Object instances[] = getApplicationLifecycleListeners();if (instances == null || instances.length == 0) {return ok;}ServletContextEvent event = new ServletContextEvent(getServletContext());ServletContextEvent tldEvent = null;if (noPluggabilityListeners.size() > 0) {noPluggabilityServletContext = new NoPluggabilityServletContext(getServletContext());tldEvent = new ServletContextEvent(noPluggabilityServletContext);}for (int i = 0; i < instances.length; i++) {// 只调用ServletContextListener// spring的ContextLoaderListener就是// ServletContextListener的实现类if (!(instances[i] instanceof ServletContextListener))continue;ServletContextListener listener =(ServletContextListener) instances[i];try {fireContainerEvent("beforeContextInitialized", listener);if (noPluggabilityListeners.contains(listener)) {listener.contextInitialized(tldEvent);} else {// 调用contextInitialized方法// 可以通过event获取ServletContext// spring的ContextLoaderListener// 是在这里加载spring的root WebApplicationContet// 并作为一个attribute填充到ServletContext中listener.contextInitialized(event);}fireContainerEvent("afterContextInitialized", listener);}...}return (ok);
    }
    

4. 过滤器Filter的实例化

  • 遍历StandardContext的filterDefs集合,创建ApplicationFilterConfig,然后填充到filterConfigs中。其中在创建ApplicationFilterConfig时,会创建filter实例并进行初始化。
    /*** Configure and initialize the set of filters for this Context.* @return <code>true</code> if all filter initialization completed* successfully, or <code>false</code> otherwise.*/
    public boolean filterStart() {if (getLogger().isDebugEnabled()) {getLogger().debug("Starting filters");}// Instantiate and record a FilterConfig for each defined filterboolean ok = true;synchronized (filterConfigs) {filterConfigs.clear();for (Entry<String,FilterDef> entry : filterDefs.entrySet()) {String name = entry.getKey();if (getLogger().isDebugEnabled()) {getLogger().debug(" Starting filter '" + name + "'");}try {// 创建ApplicationFilterConfig实例,// 并放到StandardContext的filterConfigs中ApplicationFilterConfig filterConfig =new ApplicationFilterConfig(this, entry.getValue());filterConfigs.put(name, filterConfig);} catch (Throwable t) {t = ExceptionUtils.unwrapInvocationTargetException(t);ExceptionUtils.handleThrowable(t);getLogger().error(sm.getString("standardContext.filterStart", name), t);ok = false;}}}return ok;
    }/*** Construct a new ApplicationFilterConfig for the specified filter* definition.** @param context The context with which we are associated* @param filterDef Filter definition for which a FilterConfig is to be*  constructed** @exception ClassCastException if the specified class does not implement*  the <code>javax.servlet.Filter</code> interface* @exception ClassNotFoundException if the filter class cannot be found* @exception IllegalAccessException if the filter class cannot be*  publicly instantiated* @exception InstantiationException if an exception occurs while*  instantiating the filter object* @exception ServletException if thrown by the filter's init() method* @throws NamingException* @throws InvocationTargetException* @throws SecurityException* @throws NoSuchMethodException* @throws IllegalArgumentException*/
    ApplicationFilterConfig(Context context, FilterDef filterDef)throws ClassCastException, ClassNotFoundException, IllegalAccessException,InstantiationException, ServletException, InvocationTargetException, NamingException,IllegalArgumentException, NoSuchMethodException, SecurityException {super();this.context = context;this.filterDef = filterDef;// Allocate a new filter instance if necessaryif (filterDef.getFilter() == null) {getFilter();} else {this.filter = filterDef.getFilter();// 创建filter实例getInstanceManager().newInstance(filter);// 初始化filter,如添加filter配置的param等initFilter();}
    }
    

5. Servlet的实例化

  • 主要为查找配置的Servlet中onStartup > 0的serlvet,然后加载Servlet对应的类,创建servlet实例并初始化。具体为通过StandardWrapper的load方法来完成。
    /*** Load and initialize all servlets marked "load on startup" in the* web application deployment descriptor.** @param children Array of wrappers for all currently defined*  servlets (including those not declared load on startup)* @return <code>true</code> if load on startup was considered successful*/
    public boolean loadOnStartup(Container children[]) {// 筛选loadOnStartup > 0的servlet// Collect "load on startup" servlets that need to be initializedTreeMap<Integer, ArrayList<Wrapper>> map = new TreeMap<>();for (int i = 0; i < children.length; i++) {Wrapper wrapper = (Wrapper) children[i];int loadOnStartup = wrapper.getLoadOnStartup();if (loadOnStartup < 0)continue;Integer key = Integer.valueOf(loadOnStartup);ArrayList<Wrapper> list = map.get(key);if (list == null) {list = new ArrayList<>();map.put(key, list);}list.add(wrapper);}// Load the collected "load on startup" servletsfor (ArrayList<Wrapper> list : map.values()) {for (Wrapper wrapper : list) {try {// 通过StandardWrapper的load方法来完成// 类加载,实例创建,实例初始化wrapper.load();} catch (ServletException e) {getLogger().error(sm.getString("standardContext.loadOnStartup.loadException",getName(), wrapper.getName()), StandardWrapper.getRootCause(e));// NOTE: load errors (including a servlet that throws// UnavailableException from the init() method) are NOT// fatal to application startup// unless failCtxIfServletStartFails="true" is specifiedif(getComputedFailCtxIfServletStartFails()) {return false;}}}}return true;}/*** Load and initialize an instance of this servlet, if there is not already* at least one initialized instance.  This can be used, for example, to* load servlets that are marked in the deployment descriptor to be loaded* at server startup time.* <p>* <b>IMPLEMENTATION NOTE</b>:  Servlets whose classnames begin with* <code>org.apache.catalina.</code> (so-called "container" servlets)* are loaded by the same classloader that loaded this class, rather than* the classloader for the current web application.* This gives such classes access to Catalina internals, which are* prevented for classes loaded for web applications.** @exception ServletException if the servlet init() method threw*  an exception* @exception ServletException if some other loading problem occurs*/
    @Override
    public synchronized void load() throws ServletException {instance = loadServlet();if (!instanceInitialized) {initServlet(instance);}if (isJspServlet) {StringBuilder oname = new StringBuilder(getDomain());oname.append(":type=JspMonitor");oname.append(getWebModuleKeyProperties());oname.append(",name=");oname.append(getName());oname.append(getJ2EEKeyProperties());try {jspMonitorON = new ObjectName(oname.toString());Registry.getRegistry(null, null).registerComponent(instance, jspMonitorON, null);} catch( Exception ex ) {log.info("Error registering JSP monitoring with jmx " +instance);}}
    }
    

这篇关于Tomcat源码分析(四):ServletContext应用启动之核心组件初始化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/957119

相关文章

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

SpringBoot项目启动后自动加载系统配置的多种实现方式

《SpringBoot项目启动后自动加载系统配置的多种实现方式》:本文主要介绍SpringBoot项目启动后自动加载系统配置的多种实现方式,并通过代码示例讲解的非常详细,对大家的学习或工作有一定的... 目录1. 使用 CommandLineRunner实现方式:2. 使用 ApplicationRunne

将Python应用部署到生产环境的小技巧分享

《将Python应用部署到生产环境的小技巧分享》文章主要讲述了在将Python应用程序部署到生产环境之前,需要进行的准备工作和最佳实践,包括心态调整、代码审查、测试覆盖率提升、配置文件优化、日志记录完... 目录部署前夜:从开发到生产的心理准备与检查清单环境搭建:打造稳固的应用运行平台自动化流水线:让部署像

vue解决子组件样式覆盖问题scoped deep

《vue解决子组件样式覆盖问题scopeddeep》文章主要介绍了在Vue项目中处理全局样式和局部样式的方法,包括使用scoped属性和深度选择器(/deep/)来覆盖子组件的样式,作者建议所有组件... 目录前言scoped分析deep分析使用总结所有组件必须加scoped父组件覆盖子组件使用deep前言

基于Qt Qml实现时间轴组件

《基于QtQml实现时间轴组件》时间轴组件是现代用户界面中常见的元素,用于按时间顺序展示事件,本文主要为大家详细介绍了如何使用Qml实现一个简单的时间轴组件,需要的可以参考下... 目录写在前面效果图组件概述实现细节1. 组件结构2. 属性定义3. 数据模型4. 事件项的添加和排序5. 事件项的渲染如何使用

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

Linux中Curl参数详解实践应用

《Linux中Curl参数详解实践应用》在现代网络开发和运维工作中,curl命令是一个不可或缺的工具,它是一个利用URL语法在命令行下工作的文件传输工具,支持多种协议,如HTTP、HTTPS、FTP等... 目录引言一、基础请求参数1. -X 或 --request2. -d 或 --data3. -H 或

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链