Tomcat源码分析之 doGet方法(三)

简介: Tomcat源码分析之 doGet方法(三)

通过前面两讲我们知道,客户端的请求会经过一系列的过滤器,最终达到 Servlet,我们深入理解了Tomcat 过滤器的实现机制以及 Servlet 的相关处理。在了解这些之后,我们不禁要问,客户端请求在到达这一系列的过滤器之前发生了什么呢?

 

1 目标

本次源码分析目标了解客户端请求在StandardWrapperValve 中做了哪些工作。

 

2 分析方法

根据第一讲的堆栈信息,结合 Intellij Idea 的堆栈视图、断点、单步调试等手段分析源码。

9. atorg.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:199)

3 分析流程

image.png

从上图我们可以看到,客户端请求在到达过滤器之前,会经历一系列的 Valve,本讲我们重点分析 StandardWrapperValve

 

3.1 ApplicationFilterChain 的初始化

 

StandardWrapperValue invoke()方法中代码较多,但核心的代码是如何实例化 ApplicationFilterChain

 

// Create the filter chain for this request
ApplicationFilterChain filterChain =
       ApplicationFilterFactory.createFilterChain(request,wrapper, servlet);

 

通过工厂方式创建ApplicationFilterChain对象,这是一种典型的使用工厂模式来创建对象。我们在进行源码分析的时候,经常会遇到各种设计模式的运用,所以在源码分析工作开始之前,大家务必要熟练的掌握各种经典模式的概念和一般代码编写思路,这样将极大的帮助你更好更快的理解源码。

 

每一个请求到达 Tomcat 容器的时候,都会经过一系列的 Valve,即都会经过 StandardWrapperValve,而在这个地方都会实例化一个 ApplicationFilterChain 对象,因此这种对象创建的工作是十分频繁的,所以需要一种创建型的设计模式来解决这个问题。

 

接下来我们就重点来看一下工厂模式是如何创建对象的。

 

在进行源码分析的时候,我们只关注最核心的流程,其他与核心业务无关的代码,我们将忽略。

 

filterChain= (ApplicationFilterChain) req.getFilterChain();
if (filterChain == null) {
   filterChain =
new ApplicationFilterChain();
   req.setFilterChain(filterChain);
}

 

首先查看 req 是否已经有了 filterChain 对象,如果没有则直接实例化,并赋值给 req 的属性。

 

现在我们已经有了一个基本的 filterChain 对象了,接下来就需要对其一些关键属性进行赋值操作。

 

通过上一讲我们知道Tomcat 的过滤器机制设计的核心就是 filterChain,它管理了所有的过滤器,因此在初始化的时候就需要添加所有的与本请求相关的过滤器信息。这个工作也是 filterChain 初始化的核心所在。

 

3.2 filterChain添加过滤器

 

我们在进行过滤器配置的时候,首先进行过滤器名称的定义:


<
filter>
   <
filter-name>sampleFilter</filter-name>
   <
filter-class>SampleFilter</filter-class>
</
filter>

其次是过滤器作用的映射配置,而这种映射配置主要有两种方式,

第一种是 url-pattern 方式,如下所示:

<filter-mapping>
   <
filter-name>sampleFilter</filter-name>
   <
url-pattern>/sample</url-pattern>
</
filter-mapping>

第二种是 servlet-name 方式,如下所示:
<
filter-mapping>
   <
filter-name>sampleFilter</filter-name>
   <
servlet-name>helloServlet</servlet-name>
</
filter-mapping>

 

因此在filterChain 中添加过滤器时,核心就是这两种方式的实现,下面将继续介绍Tomcat 是如何实现这两种方式的

 

3.2.1 url-pattern 方式

 

基于 url-pattern 的方式添加过滤器:

// Add the relevant path-mapped filters to this filter chain
for (int i = 0; i < filterMaps.length; i++) {
    if (!matchDispatcher(filterMaps[i],dispatcher)) {
        continue;
    }
    if (!matchFiltersURL(filterMaps[i],requestPath))
        continue;
    ApplicationFilterConfig filterConfig= (ApplicationFilterConfig)
       context.findFilterConfig(filterMaps[i].getFilterName());
    if (filterConfig == null) {
        // FIXME - logconfiguration problem
        continue;
    }
    filterChain.addFilter(filterConfig);
}

核心方法是matchFiltersURL(),将所有符合请求路径的过滤器添加进来。

private static boolean matchFiltersURL(String testPath, StringrequestPath) {
    if (testPath == null)
        return false;
    // Case 1 - Exact Match
    if (testPath.equals(requestPath))
        return true;
    // Case 2 - Path Match("/.../*")
    if (testPath.equals("/*"))
        return true;
    if (testPath.endsWith("/*")) {
        if (testPath.regionMatches(0, requestPath, 0,
                                  testPath.length() - 2)) {
            if (requestPath.length() ==(testPath.length() - 2)){
                return true;
            } else if ('/' ==requestPath.charAt(testPath.length() - 2)) {
                return true;
            }
        }
        return false;
    }
    // Case 3 - Extension Match
    if (testPath.startsWith("*.")) {
        int slash = requestPath.lastIndexOf('/');
        int period = requestPath.lastIndexOf('.');
        if ((slash >= 0) && (period > slash)
            && (period !=requestPath.length() - 1)
            &&((requestPath.length() - period)
                == (testPath.length() - 1))) {
            return (testPath.regionMatches(2, requestPath, period + 1,
                                           testPath.length() - 2));
        }
    }
    // Case 4 - "Default" Match
    return false; // NOTE - Not relevantfor selecting filters
}

以上就是四种情况的匹配,不再详细描述,感兴趣的同学可以深入了解,后面我们会再安排文章介绍。

 

3.2.2 servlet-name 方式

// Add filters that match on servlet name second
for (int i = 0; i < filterMaps.length; i++) {
    if (!matchDispatcher(filterMaps[i],dispatcher)) {
        continue;
    }
    if (!matchFiltersServlet(filterMaps[i],servletName))
        continue;
    ApplicationFilterConfig filterConfig= (ApplicationFilterConfig)
       context.findFilterConfig(filterMaps[i].getFilterName());
    if (filterConfig == null) {
        // FIXME - logconfiguration problem
        continue;
    }
    filterChain.addFilter(filterConfig);
}
核心方法是 matchFiltersServlet(),如下所示:
private static boolean matchFiltersServlet(FilterMap filterMap,
                                   String servletName) {
    if (servletName == null) {
        return false;
    }
    // Check the specific "*"special servlet name
    else if (filterMap.getMatchAllServletNames()) {
        return true;
    } else {
        String[] servletNames =filterMap.getServletNames();
        for (int i = 0; i < servletNames.length; i++) {
            if (servletName.equals(servletNames[i])){
                return true;
            }
        }
        return false;
    }
}

获得所有过滤器的 servletName,然后遍历匹配到当前 servletName 的过滤器并添加到 filterChain 中。

 

这里面大家需要注意的是,以上两种方式添加的过滤器有可能会重复,因此 filterChain 在添加的时候也会做重复性添加的检查,另外如果数组容量不够时也会对数组进行扩容操作,具体代码如下。

void addFilter(ApplicationFilterConfig filterConfig) {
    // Prevent the same filter being addedmultiple times
    for(ApplicationFilterConfig filter:filters)
        if(filter==filterConfig)
            return;
    if (n == filters.length) {
        ApplicationFilterConfig[]newFilters =
            new ApplicationFilterConfig[n + INCREMENT];
        System.arraycopy(filters, 0, newFilters, 0, n);
        filters = newFilters;
    }
    filters[n++] = filterConfig;
}

 

还有一点需要注意的是,在 ApplicationFilterChain 中保存的数组并不是 Filter[],而是ApplicationFilterConfig[]ApplicationFilterConfig 类实现了 FilterConfig 接口,在这个接口中保存了有关过滤器的名字、初始化参数等所有信息,其定义如下所示:

public interface FilterConfig {
    public String getFilterName();
    public ServletContext getServletContext();
    public String getInitParameter(String name);
    public Enumeration<String> getInitParameterNames();
}

4 总结

 

本讲重点探讨了客户端请求在到达过滤器之前,会经过一系列的 Valve,重点分析了StandardWrapperValve 的执行流程,尤其是 ApplicationFilterChain 的初始化工作。

 

下一讲我们将分析 StandardContextValve 组件。

目录
相关文章
|
3月前
|
监控 Java 应用服务中间件
Spring Boot整合Tomcat底层源码分析
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置和起步依赖等特性,大大简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是其与Tomcat的整合。
85 1
|
5月前
|
监控 网络协议 应用服务中间件
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
本文详细解析了Tomcat架构中复杂的`Connector`组件。作为客户端与服务器间沟通的桥梁,`Connector`负责接收请求、封装为`Request`和`Response`对象,并传递给`Container`处理。文章通过四个关键问题逐步剖析了`Connector`的工作原理,并深入探讨了其构造方法、`init()`与`start()`方法。通过分析`ProtocolHandler`、`Endpoint`等核心组件,揭示了`Connector`初始化及启动的全过程。本文适合希望深入了解Tomcat内部机制的读者。欢迎关注并点赞,持续更新中。如有问题,可搜索【码上遇见你】交流。
【Tomcat源码分析】从零开始理解 HTTP 请求处理 (第一篇)
|
5月前
|
人工智能 前端开发 Java
【Tomcat源码分析】启动过程深度解析 (二)
本文深入探讨了Tomcat启动Web应用的过程,重点解析了其加载ServletContextListener及Servlet的机制。文章从Bootstrap反射调用Catalina的start方法开始,逐步介绍了StandardServer、StandardService、StandardEngine、StandardHost、StandardContext和StandardWrapper的启动流程。每个组件通过Lifecycle接口协调启动,子容器逐层启动,直至整个服务器完全启动。此外,还详细分析了Pipeline及其Valve组件的作用,展示了Tomcat内部组件间的协作机制。
【Tomcat源码分析】启动过程深度解析 (二)
|
4月前
apache+tomcat配置多站点集群的方法
apache+tomcat配置多站点集群的方法
69 4
|
5月前
|
前端开发 Java 应用服务中间件
【Tomcat源码分析 】"深入探索:Tomcat 类加载机制揭秘"
本文详细介绍了Java类加载机制及其在Tomcat中的应用。首先回顾了Java默认的类加载器,包括启动类加载器、扩展类加载器和应用程序类加载器,并解释了双亲委派模型的工作原理及其重要性。接着,文章分析了Tomcat为何不能使用默认类加载机制,因为它需要解决多个应用程序共存时的类库版本冲突、资源共享、类库隔离及JSP文件热更新等问题。最后,详细展示了Tomcat独特的类加载器设计,包括Common、Catalina、Shared、WebApp和Jsp类加载器,确保了系统的稳定性和安全性。通过这种设计,Tomcat实现了不同应用程序间的类库隔离与共享,同时支持JSP文件的热插拔。
【Tomcat源码分析 】"深入探索:Tomcat 类加载机制揭秘"
|
5月前
|
设计模式 应用服务中间件 容器
【Tomcat源码分析】Pipeline 与 Valve 的秘密花园
本文深入剖析了Tomcat中的Pipeline和Valve组件。Valve作为请求处理链中的核心组件,通过接口定义了关键方法;ValveBase为其基类,提供了通用实现。Pipeline则作为Valve容器,通过首尾相连的Valve链完成业务处理。StandardPipeline实现了Pipeline接口,提供了详细的Valve管理逻辑。通过对代码的详细分析,揭示了模板方法模式和责任链模式的应用,展示了系统的扩展性和模块化设计。
【Tomcat源码分析】Pipeline 与 Valve 的秘密花园
|
5月前
|
设计模式 人工智能 安全
【Tomcat源码分析】生命周期机制 Lifecycle
Tomcat内部通过各种组件协同工作,构建了一个复杂的Web服务器架构。其中,`Lifecycle`机制作为核心,管理组件从创建到销毁的整个生命周期。本文详细解析了Lifecycle的工作原理及其方法,如初始化、启动、停止和销毁等关键步骤,并展示了LifecycleBase类如何通过状态机和模板模式实现这一过程。通过深入理解Lifecycle,我们可以更好地掌握组件生命周期管理,提升系统设计能力。欢迎关注【码上遇见你】获取更多信息,或搜索【AI贝塔】体验免费的Chat GPT。希望本章内容对你有所帮助。
|
6月前
|
网络协议 Java 应用服务中间件
Tomcat源码分析 (一)----- 手撕Java Web服务器需要准备哪些工作
本文探讨了后端开发中Web服务器的重要性,特别是Tomcat框架的地位与作用。通过解析Tomcat的内部机制,文章引导读者理解其复杂性,并提出了一种实践方式——手工构建简易Web服务器,以此加深对Web服务器运作原理的认识。文章还详细介绍了HTTP协议的工作流程,包括请求与响应的具体格式,并通过Socket编程在Java中的应用实例,展示了客户端与服务器间的数据交换过程。最后,通过一个简单的Java Web服务器实现案例,说明了如何处理HTTP请求及响应,强调虽然构建基本的Web服务器相对直接,但诸如Tomcat这样的成熟框架提供了更为丰富和必要的功能。
|
6月前
|
Ubuntu Java 应用服务中间件
在Ubuntu 16.04上安装Apache Tomcat 8的方法
在Ubuntu 16.04上安装Apache Tomcat 8的方法
81 0