Tomcat处理请求的流程

 更新时间:2023年05月26日 11:11:38   作者:单手入天象  
本文主要介绍了Tomcat处理请求的流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
(福利推荐:【腾讯云】服务器最新限时优惠活动,云服务器1核2G仅99元/年、2核4G仅768元/3年,立即抢购>>>:9i0i.cn/qcloud

(福利推荐:你还在原价购买阿里云服务器?现在阿里云0.8折限时抢购活动来啦!4核8G企业云服务器仅2998元/3年,立即抢购>>>:9i0i.cn/aliyun

一、组件详解

在Tomcat处理客户端请求的过程中,这里面有三个组件概念,他们都是线程,分别负责不同的职责。
(必须记清楚这三个线程组件)

Acceptor

一个普通线程任务,用于接收新连接,并将新连接封装,选择一个 Poller 将新连接添加到 Poller 的事件队列中。

Poller

一个线程任务,用于监听 Socket 事件,当有任务来临时,将 Socket 封装,添加到 worker 线程池的任务队列中。

Worker

他是创建一组线程的,每个线程任务都是一个阻塞队列,用于对请求进行处理(例如我们中间件参数中的最大线程数就是指最多创建多少个Worker线程)。每个Worker线程任务包括分析解析请求报文、创建 Request 对象、调用容器的 管道pipeline 执行阀门value、执行servlet的具体逻辑。

二、请求处理流程

1.总体流程图

2.Worker线程任务流程

三、源码跟踪

1.Tomcat启动线程组件

Tomcat启动时,如果默认使用NIO模式,先是执行了AbstractEndpoint.initServerSocket,通过 ServerSocketChannel.open() 打开一个 ServerSocket通道,默认绑定到 8080 端口,用于监听请求。

说明:在Java语言的NIO中,类ServerSocketChannel就是用来处理TPC连接的客户端,他的open方法就是用例建立一个TPC连接。

然后Tomcat会创建Worker 线程池、Acceptor线程、Poller线程:

AbstractEndpoint.createExecutor,用于创建 Worker 线程池,这个线程池是用来处理实际的请求的,把配置文件中的初始线程数10、最大线程数200等信息传进去,创建一个线程池executor

AbstractEndpoint.createExecutor.startAcceptorThread,他创建一个线程任务Acceptor,作为一个接收者,线程用来无限循环接受客户端发送过来的连接请求

NioEndpoint.startInternal,创建一个线程任务Poller,用于检测Acceptor接兽并处理成已就绪的 Socket。

2.Acceptor

Tomcat启动完成后,客户端发起一个请求

Acceptor的run方法,无限循环在这里接受连接请求
(假如启动后客户端发起一个请求,这里就是第一时间捕获到)

点进去NioEndpoint.serverSocketAccept(因为使用NIO模式),可以看到我们熟悉的nio的accept方法,这是一个阻塞的方法,会一直等待接收请求。

当Acceptor接收到客户端的请求时,调用addEvent() 方法会将 Socket 添加到该 Poller 的 PollerEvent 队列中。并调用了NIO中selector.wakeup方法,唤醒了Poller。到此,这一次请求中 Acceptor 的任务就完成了。

3.Poller

接着到Poller 线程了,Poller 线程1秒阻塞一次,等待有请求过来被唤醒后,每次请求先过AbstractEndpoint.processSocket

从处理器缓存中获取当前要被执行的任务,放进任务进程,然后获取Worker线程组,将这个任务放进去。到此 Poller 的任务就完成了。

4.Worker

然后就是到Worker线程组了,这次请求的后续的所有操作都在这个线程中完成。Worker线程是一个阻塞队列,它继承自AbstractQueuedSynchronizer。worker 线程被创建以后就执行 ThreadPoolExecutor 的 runWorker() 方法,试图从 workQueue 中取待处理任务,但是一开始 workQueue 是空的,所以 worker 线程会阻塞在 workQueue.take() 方法。

当新任务添加到 workQueue后,workQueue.take() 阻塞就会结束,会返回一个 Runnable,通常是 SocketWrapperBase,然后 worker 线程调用 SocketWrapperBase的 run() 方法对 Socket 进行处理。

执行SocketWrapperBase.run

里面调用的是 doRun方法,他是抽象方法,根据当前Tomcat使用的模式是NIO还是APR去选择执行不同的方法(默认是NIO执行NioEndpoint里的内部类SocketProcessor.doRun)

这个socket处理器先做TPC的三次握手

三次握手,这里Tomcat作为服务端,是需要响应(执行)两次的,源码断点发现每次http请求这里都是执行两次

三次握手中该方法执行两次:

第一次执行时event对象是null,执行完是OPEN_READ,表示数据可供客户端读取

第二次执行时event对象是OPEN_READ,如果停用长连接,执行完返回的是CLOSE,关闭连接,否则不关闭
走到下面,他是获取协议处理器,并执行他的process方法

执行AbstractProtocol.process

可以看到他是获取的Http11Processor,因为默认用的协议是http1.1

下面执行这个处理器的process方法

跟进去AbstractProcessorLight.process,然后到了Http11Processor.service,service方法先是从当前请求request中,解析请求行、请求头、请求体,封装成Request对象

下面获取adapter(CoyoteAdaptor),调用service方法

执行CoyoteAdaptor.service

获取到Request和Response并封装

然后调用postParseRequest方法,在 Mapper 中查询 URL 的映射关系

下面把封装成的Request对象和响应的Response对象传递给Engine容器,然后获取他的管道,执行里面绑定的阀门value

按顺序执行多个阀门,实现对应的功能

最后执行到StandardWrapperValve.invoke

将Servlet封装到FilterChain过滤器链中

他是定位到ApplicationFilterChain.doFilter,里面先是执行了Tomcat内置的过滤器

下面执行了servlet.service

然后这里就是去调用我们熟悉的HttpServlet的service方法,解析里面对应的doGet方法,或者doPost方法等等… ,也就是执行具体业务方法。

最后由Servlet将响应返回给了客户端。

总结

到此这篇关于Tomcat处理请求的流程的文章就介绍到这了,更多相关Tomcat处理请求内容请搜索程序员之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持程序员之家!

相关文章

  • Tomcat启动时报错:java.lang.UnsatisfiedLinkError的解决

    Tomcat启动时报错:java.lang.UnsatisfiedLinkError的解决

    这篇文章主要跟大家介绍了在Tomcat启动时报错:java.lang.UnsatisfiedLinkError的解决方法,虽然这个错误不影响项目的启动运行,但是有强迫症的程序员会心里不爽,下面来一起看看解决的方法吧。
    2017-06-06
  • Mac环境下配置tomcat的步骤详解

    Mac环境下配置tomcat的步骤详解

    我们在MAC系统中查看网页时,一般都要使用到tomcat,这是因为appache只支持静态网页,但像asp,php,cgi,jsp等动态就需要tomcat来处理。那么该怎么在自己的MAC中安装tomcat呢?现在小编就教大家安装的方法,有需要的朋友们可以参考借鉴。
    2016-10-10
  • Tomcat中使用ipv6地址的示例代码

    Tomcat中使用ipv6地址的示例代码

    在公司的一次项目改造过程中,需要将原来的IPV6替换成IPV4,本文就详细的介绍Tomcat中使用ipv6地址的示例代码,具有一定的参考价值,感兴趣的可以了解一下
    2022-05-05
  • 解读Tomcat启动、重启、暂停操作(window)

    解读Tomcat启动、重启、暂停操作(window)

    这篇文章主要介绍了解读Tomcat启动、重启、暂停操作(window),具有很好的参考价值,希望对大家有所帮助。
    2023-04-04
  • tomcat点击startup.bat出现闪退的原因及解决方法

    tomcat点击startup.bat出现闪退的原因及解决方法

    本文主要介绍了tomcat点击startup.bat出现闪退的原因及解决方法,文中通过图文介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2023-09-09
  • intellij idea 使用Tomcat部署的项目位置在哪

    intellij idea 使用Tomcat部署的项目位置在哪

    intellij idea 使用Tomcat部署的项目在哪里,为什么不在Tomcat的webapps目录下面,本文通过图文实例相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2021-01-01
  • Java开启/关闭tomcat服务器的方法

    Java开启/关闭tomcat服务器的方法

    这篇文章主要介绍了Java开启/关闭tomcat服务器的方法,本文给大家介绍的非常详细,具有一定的参考借鉴价值 ,需要的朋友可以参考下
    2019-06-06
  • tomcat服务器安全设置方法

    tomcat服务器安全设置方法

    这篇文章主要介绍了tomcat服务器安全设置方法,需要的朋友可以参考下
    2019-11-11
  • Tomcat容器管理安全的验证方式汇总

    Tomcat容器管理安全的验证方式汇总

    当访问服务器中受保护的资源时,容器管理的验证方法可以控制确认用户身份的方式。Tomcat支持四种容器管理的安全防护。下面小编给大家分享Tomcat容器管理安全的验证方式汇总,感兴趣的朋友一起看看吧
    2016-11-11
  • Tomcat请求处理流程与源码浅析(最新推荐)

    Tomcat请求处理流程与源码浅析(最新推荐)

    这篇文章主要介绍了Tomcat请求处理流程与源码浅析,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-05-05

最新评论

?


http://www.vxiaotou.com