一: tomcat系统架构
如上,tomcat主要分为两部分
- http服务器: 处理http请求,将字节流的request信息封装为servlet能实别的ServletRequest,然后交给servlet处理。 当返回ServletResponse后又处理为Response对象给客户端。
- servlet容器:根据url匹配servlet,最终调用业务类处理逻辑。
对于上面两个部分,tomcat设计了两大组件分别实现核心功能: 连接器(Connector)和容器(Container)
- Connector: 核心组件Coyote,处理Socket请求,并与Catalina打交道。也就说Coyote处理应用层的Http协议和传输层的NIO模型(tomcat8前默认BIO,现在默认NIO)
- Catalina:servlet容器部分的核心实现,一个tomcat就是一个Catalina实例,对应一个server实例(server实例有Catalina根据server.xml创建)
如图示:
一个server包含多个service,service把多个Connector组件绑定到一个Container。 而Container容器负责处理⽤户的servlet请求,具体来看一下server.xml配置:
<!--port:关闭服务器的监听端⼝ shutdown:关闭服务器的指令字符串-->
<Server port="8005" shutdown="SHUTDOWN"><Listener className="org.apache.catalina.startup.VersionLoggerListener" />
<Listener className="org.apache.catalina.core.AprLifecycleListener" SSLEngine="on" />
<Listener className="org.apache.catalina.core.JreMemoryLeakPreventionListener" />
<Listener className="org.apache.catalina.mbeans.GlobalResourcesLifecycleListener" />
<Listener className="org.apache.catalina.core.ThreadLocalLeakPreventionListener" /><GlobalNamingResources>
<Resource name="UserDatabase" auth="Container"
type="org.apache.catalina.UserDatabase"
description="User database that can be updated and saved"
factory="org.apache.catalina.users.MemoryUserDatabaseFactory"
pathname="conf/tomcat-users.xml" />
</GlobalNamingResources>
<!--
该标签用于创建 Service 实例,默认使 org.apache.catalina.core.StandardService。
默认情况下,Tomcat 仅指定了Service 的名称, 值为 "Catalina"。
Service ⼦标签为 : Listener、Executor、Connector、Engine,
其中:
Listener 用于为Service添加⽣命周期监听器,
Executor 用于配置Service 共享线程池,
Connector 用于配置Service 包含的链接器,
Engine 用于配置Service中链接器对应的Servlet 容器引擎
-->
<Service name="Catalina">
<--
service下多个Connector的共享线程池,默认不创建。 如果我们想添加一个线程池, 可以在<Service> 下添加如下配置:
name:线程池名称,⽤于 Connector中指定
namePrefix:所创建的每个线程的名称前缀,⼀个单独的线程名称为
namePrefix+threadNumber
maxThreads:池中最⼤线程数
minSpareThreads:活跃线程数,也就是核⼼池线程数,这些线程不会被销毁,会⼀直存在
maxIdleTime:线程空闲时间,超过该时间后,空闲线程会被销毁,默认值为6000(1分钟),单位毫秒
maxQueueSize:在被执⾏前最⼤线程排队数⽬,默认为Int的最⼤值,也就是⼴义的⽆限。除特殊情况,这个值 不需要更改,否则会有请求不会被处理的情况
prestartminSpareThreads:启动线程池时是否启动 minSpareThreads部分线程。默认值为false,即不启动
threadPriority:线程池中线程优先级,默认值为5,值从1到10
className:线程池实现类,未指定情况下,默认实现类为
org.apache.catalina.core.StandardThreadExecutor。如果想使⽤⾃定义线程池⾸先需要实现
org.apache.catalina.Executor接⼝
--><!--线程池配置,默认注释-->
<Executor name="commonThreadPool"
namePrefix="thread-exec-"
maxThreads="200"
minSpareThreads="100"
maxIdleTime="60000"
maxQueueSize="Integer.MAX_VALUE"
prestartminSpareThreads="false"
threadPriority="5"
className="org.apache.catalina.core.StandardThreadExecutor"/>
<!--Connector标签定义端口,域名在下面的HOST标签定义-->
<!--处理http协议,定义监听端口。如果是加密的https,转发到8443端口-->
<Connector port="8080" protocol="HTTP/1.1"
connectionTimeout="20000"
redirectPort="8443" enableLookups="false" />
<!--处理AJP协议,不能处理也转发到8443端口-->
<Connector port="8009" protocol="AJP/1.3" redirectPort="8443" />
<!--一个service下对应一个Engine,真正处理请求的,默认虚拟机为下面第二个Host(没有能处理的Host时,选用默认虚拟主机)-->
<Engine name="Catalina" defaultHost="localhost">
<Realm className="org.apache.catalina.realm.LockOutRealm">
<Realm className="org.apache.catalina.realm.UserDatabaseRealm"
resourceName="UserDatabase" />
</Realm><!-- 接收域名为www.longshuai.com的请求,此应用的路径在tomcat目录的webapps下 -->
<Host name="www.longshuai.com" appBase="webapps"
unpackWARs="true" autoDeploy="true">
<!--通常一个tomcat下放一个应用程序,默认在webapps下; 但是呢,一个tomcat也可以加载多个web应用,可以放到不同的目录下,用不同的Context标签代表不同的应用-->
<!--web应用1,应用放在绝对路径/www/longshuai路径下,访问url为:www.longshuai.com:8080-->
<Context path="" docBase="/www/longshuai" reloadable="true" />
<!--web应用2,应用放在appBase的相对路径下,也就是webapps/xuexi,访问url为:www.longshuai.com:8080/xuexi-->
<Context path="/xuexi" docBase="xuexi" reloadable="true" />
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="longshuai_access_log" suffix=".txt"
pattern="%h %l %u %t "%r" %s %b" />
</Host><!-- 默认虚拟主机localhost,可以不修改 -->
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="true">
<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
prefix="localhost_access_log" suffix=".txt"
pattern="%h %l %u %t "%r" %s %b" />
</Host>
</Engine>
</Service>
</Server>
附:多个service一般是对于同名的多程序部署:Tomcat配置多个Service_寒小二的博客-CSDN博客_tomcat配置多个service)
二:tomcat调优
系统调优一方面是jvm自身调优,现在介绍tomcat调休:
- 可以在service中使用共享线程池。
- 禁⽤ AJP 连接器,就是注释掉8009那个协议
- AJP是一个长连接,避免每次心情求创建新连接。但是AJP服务只有Apache服务器才能使用,而我们一般是使用Nginx + tomcat的架构,所以用不着AJP协议,所以把AJP连接器禁用;换个角度想,如果一个AJP服务我们用不到但是是开启着,肯定会造成资源的浪费。
- 使用NIO: 将红色方框的内容替换为黄色的部分
-
Tomcat8 之前的版本默认使⽤ BIO (阻塞式 IO),对于每⼀个请求都要创建⼀个线程来处理,不适合⾼并发; Tomcat8 以后的版本默认使⽤ NIO 模式(⾮阻塞式 IO )
-
protocol="org.apache.coyote.http11.Http11NioProtocol"
- 动静分离
- 静态资源存到nginx
- 调整一些参数,如:
-
maxConnections: 最大连接数。 每个连接最终会交个一个线程处理,无可用线程也可以继续接收连接,但肯定需要设置一个合适的值,根据cpu情况设置(一般还是要压测)
-
maxThreads: 最大线程数。
-
acceptCount:等待连接数。 达到maxConnections后,后续的连接会放到队列中等待。
-
tomcat其他调优