How Tomcat Works 3

2024-03-15 00:32
文章标签 tomcat works

本文主要是介绍How Tomcat Works 3,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Tomcat体系结构中有一个非常重要的概念——连接器,看下面这张图


一个connecotr对应一个container,connector用于接收请求,然后传递给container组件去处理,tomcat中默认的连接器是coyote,按照协议类型分,有http和ajp的连接器。

HTTP/1.1协议负责建立HTTP连接,web应用通过浏览器访问tomcat服务器用的就是这个连接器,默认监听的是8080端口;

AJP/1.3协议负责和其他HTTP服务器建立连接,监听的是8009端口,比如tomcat和apache或者iis集成时需要用到这个连接器。

下面是一个简单的connector实现:

package server2.connector;import java.io.IOException;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.UnknownHostException;
/*** HttpConnector主要负责接入消息,分发给不同的处理器* * @author Administrator**/
public class HttpConnector implements Runnable{private boolean shutdown = false;// 服务器是否停止private String scheme = "http";public String getScheme() {return scheme;}public void setScheme(String scheme) {this.scheme = scheme;}@Overridepublic void run() {ServerSocket serverSocket = null;int port = 8080;try {serverSocket = new ServerSocket(port, 1,InetAddress.getByName("127.0.0.1"));} catch (UnknownHostException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}Socket socket = null;while (!shutdown) {// 不停的接受请求try {socket = serverSocket.accept();} catch (IOException e) {e.printStackTrace();}HttpProcessor processor = new HttpProcessor();processor.process(socket);}}/*** 启动connector监听线程*/public void start(){System.out.println("started!!!");Thread thread = new Thread(this);thread.start();}
}
这是一种非常原始的处理方法,一旦接收到请求便会构造出一个处理器,由处理器去处理socket,并且process是一个同步的方法,这里没有用到Processor的池化,也没有使用异步的方式处理socket。这种同步阻塞的方式,connector的处理能力有限,一旦处理过程被阻塞,那么connector便会拒绝下一个请求。

异步方式的connector它的run方法如下:

/*** The background thread that listens for incoming TCP/IP connections and* hands them off to an appropriate processor.*/public void run() {// Loop until we receive a shutdown commandwhile (!stopped) {// Accept the next incoming connection from the server socketSocket socket = null;try {//                if (debug >= 3)//                    log("run: Waiting on serverSocket.accept()");//接收客户端的socket连接socket = serverSocket.accept();//                if (debug >= 3)//                    log("run: Returned from serverSocket.accept()");if (connectionTimeout > 0)socket.setSoTimeout(connectionTimeout);socket.setTcpNoDelay(tcpNoDelay);} catch (AccessControlException ace) {log("socket accept security exception", ace);continue;} catch (IOException e) {//                if (debug >= 3)//                    log("run: Accept returned IOException", e);try {// If reopening fails, exitsynchronized (threadSync) {if (started && !stopped)log("accept error: ", e);if (!stopped) {//                    if (debug >= 3)//                        log("run: Closing server socket");serverSocket.close();//                        if (debug >= 3)//                            log("run: Reopening server socket");serverSocket = open();}}//                    if (debug >= 3)//                        log("run: IOException processing completed");} catch (IOException ioe) {log("socket reopen, io problem: ", ioe);break;} catch (KeyStoreException kse) {log("socket reopen, keystore problem: ", kse);break;} catch (NoSuchAlgorithmException nsae) {log("socket reopen, keystore algorithm problem: ", nsae);break;} catch (CertificateException ce) {log("socket reopen, certificate problem: ", ce);break;} catch (UnrecoverableKeyException uke) {log("socket reopen, unrecoverable key: ", uke);break;} catch (KeyManagementException kme) {log("socket reopen, key management problem: ", kme);break;}continue;}// Hand this socket off to an appropriate processor//创建processor,如果连接池没有可用的处理器,则拒绝消息,其中的createProcessor方法基于栈来实现的处理池HttpProcessor processor = createProcessor();if (processor == null) {try {log(sm.getString("httpConnector.noProcessor"));socket.close();} catch (IOException e) {;}continue;}//            if (debug >= 3)//                log("run: Assigning socket to processor " + processor);//否则的话将socket传递给processorprocessor.assign(socket);// The processor will recycle itself when it finishes}// Notify the threadStop() method that we have shut ourselves down//        if (debug >= 3)//            log("run: Notifying threadStop() that we have shut down");synchronized (threadSync) {threadSync.notifyAll();}}
接收到socket调用processor的assign方法传递给HttpProcessor。处理器的实现也和上面有很大的差别,这里单独实现为一个线程类,先不急看assign方法,看看processor的run方法:

 /*** The background thread that listens for incoming TCP/IP connections and* hands them off to an appropriate processor.*/public void run() {// Process requests until we receive a shutdown signal//不停的循环处理消息while (!stopped) {// Wait for the next socket to be assigned//调用自身的await()Socket socket = await();//未获取到socket继续循环if (socket == null)continue;// Process the request from this sockettry {//获取到则进行处理process(socket);} catch (Throwable t) {log("process.invoke", t);}// Finish up this request//将该处理线程放回池中复用connector.recycle(this);}// Tell threadStop() we have shut ourselves down successfullysynchronized (threadSync) {threadSync.notifyAll();}}
await方法:

/*** Await a newly assigned Socket from our Connector, or <code>null</code>* if we are supposed to shut down.该方法是个同步方法,并且内部可能发生阻塞*/private synchronized Socket await() {// Wait for the Connector to provide a new Socket
//如果没有可处理的socket则同步阻塞,这里connector调用processor的assign方法会传递
//过来可用的socketwhile (!available) {try {wait();} catch (InterruptedException e) {}}// Notify the Connector that we have received this Socket
//否则可以处理,成员变量赋值给私有变量Socket socket = this.socket;
//设置为没有可处理的socketavailable = false;
//唤醒被阻塞的线程notifyAll();if ((debug >= 1) && (socket != null))log("  The incoming request has been awaited");return (socket);}


再来看下assign方法:

synchronized void assign(Socket socket) {// Wait for the Processor to get the previous Socket//available表示是否有可处理的socket,第一次值为false跳过while循环while (available) {//try {wait();} catch (InterruptedException e) {}}// Store the newly available Socket and notify our thread//将socket赋值给成员变量socketthis.socket = socket;//修改值为trueavailable = true;//唤醒阻塞在该对象上的线程notifyAll();if ((debug >= 1) && (socket != null))log(" An incoming request is being assigned");}
如果当前线程有可处理的socket则阻塞住connector,否则唤醒阻塞在await上的线程,处理可用的socket,这里的assign方法是异步返回的,只是负责传递socket,处理过程由processor线程完成。connector和processor线程的通讯由available变量和wait,notifyAll方发共同完成。



这篇关于How Tomcat Works 3的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Tomcat版本与Java版本的关系及说明

《Tomcat版本与Java版本的关系及说明》:本文主要介绍Tomcat版本与Java版本的关系及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Tomcat版本与Java版本的关系Tomcat历史版本对应的Java版本Tomcat支持哪些版本的pythonJ

tomcat多实例部署的项目实践

《tomcat多实例部署的项目实践》Tomcat多实例是指在一台设备上运行多个Tomcat服务,这些Tomcat相互独立,本文主要介绍了tomcat多实例部署的项目实践,具有一定的参考价值,感兴趣的可... 目录1.创建项目目录,测试文China编程件2js.创建实例的安装目录3.准备实例的配置文件4.编辑实例的

如何将Tomcat容器替换为Jetty容器

《如何将Tomcat容器替换为Jetty容器》:本文主要介绍如何将Tomcat容器替换为Jetty容器问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Tomcat容器替换为Jetty容器修改Maven依赖配置文件调整(可选)重新构建和运行总结Tomcat容器替

Centos环境下Tomcat虚拟主机配置详细教程

《Centos环境下Tomcat虚拟主机配置详细教程》这篇文章主要讲的是在CentOS系统上,如何一步步配置Tomcat的虚拟主机,内容很简单,从目录准备到配置文件修改,再到重启和测试,手把手带你搞定... 目录1. 准备虚拟主机的目录和内容创建目录添加测试文件2. 修改 Tomcat 的 server.X

Tomcat的下载安装与使用教程

《Tomcat的下载安装与使用教程》本文介绍了Tomcat的下载、安装和使用方法,包括在本机和云服务器上部署Tomcat的过程,以及解决启动失败问题的方法... 目录Tomcat的下载安装与使用Tomcat的下载与安装Tomcat在本机运行使用Tomcat在php云服务器上的使用总结Tomcat的下载安装与

Tomcat高效部署与性能优化方式

《Tomcat高效部署与性能优化方式》本文介绍了如何高效部署Tomcat并进行性能优化,以确保Web应用的稳定运行和高效响应,高效部署包括环境准备、安装Tomcat、配置Tomcat、部署应用和启动T... 目录Tomcat高效部署与性能优化一、引言二、Tomcat高效部署三、Tomcat性能优化总结Tom

通过prometheus监控Tomcat运行状态的操作流程

《通过prometheus监控Tomcat运行状态的操作流程》文章介绍了如何安装和配置Tomcat,并使用Prometheus和TomcatExporter来监控Tomcat的运行状态,文章详细讲解了... 目录Tomcat安装配置以及prometheus监控Tomcat一. 安装并配置tomcat1、安装

使用TomCat,service输出台出现乱码的解决

《使用TomCat,service输出台出现乱码的解决》本文介绍了解决Tomcat服务输出台中文乱码问题的两种方法,第一种方法是修改`logging.properties`文件中的`prefix`和`... 目录使用TomCat,service输出台出现乱码问题1解决方案问题2解决方案总结使用TomCat,

最新版IDEA配置 Tomcat的详细过程

《最新版IDEA配置Tomcat的详细过程》本文介绍如何在IDEA中配置Tomcat服务器,并创建Web项目,首先检查Tomcat是否安装完成,然后在IDEA中创建Web项目并添加Web结构,接着,... 目录配置tomcat第一步,先给项目添加Web结构查看端口号配置tomcat    先检查自己的to

Apache Tomcat服务器版本号隐藏的几种方法

《ApacheTomcat服务器版本号隐藏的几种方法》本文主要介绍了ApacheTomcat服务器版本号隐藏的几种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需... 目录1. 隐藏HTTP响应头中的Server信息编辑 server.XML 文件2. 修China编程改错误