Tomcat学习笔记(一)

一.Connector

在使用tomcat时,经常会遇到连接数、线程数之类的配置问题,要真正理解这些概念,必须先了解Tomcat的连接器(Connector)。
Connector在server.xml配置文件中,它的主要功能是接收客户端连接请求,然后创建request和response对象,
并分配线程让Engine(Servlet容器)来处理这个请求,并将刚创建的request和response对象传给Engine。
Engine处理完请求后,也将通过Connector将响应返回给客户端。
所以,Servlet容器处理请求,需要Connector进行调度和控制的,Connector是Tomcat处理请求的主干,
因此Connector的配置和使用,对Tomcat的性能有着重要的影响。
根据协议的不同,Connector可以分为HTTP Connector、AJP Connector等

二.HTTP Connector

1.Connector的protocol

Connector在处理Http请求时,会使用不同的protocol。不同tomcat版本支持的protocol不同。
其中最典型的protocol包括BIO、NIO和APR(注:Tomcat7中支持这3种,Tomcat8增加了对NIO2的支持,
而到了Tomcat8.5和Tomcat9.0,则去掉了对BIO的支持)。
BIO是Blocking IO,阻塞的IO;
NIO是Non-blocking IO,非阻塞的IO;
APR是Apache Portable Runtime,Apache可移植运行库,利用本地库可实现高扩展性,高性能;
Tomcat上运行高并发应用的首选模式时APR,但是需要安装apr,apr-utils,tomcat-native等包。

2.如何指定protocol

<Connector port="8080" protocol="HTTP/1.1"
           connectionTimeout="20000"
           redirectPort="8443" />

如果没有指定protocol,则使用默认值HTTP/1.1;
如果在tomcat7下使用HTTP/1.1,将使用BIO或APR(如果找到了APR需要的本地库,则使用APR,反之,使用BIO);
如果在tomcat8中使用HTTP/1.1,将使用NIO或APR(如果找到了APR需要的本地库,则使用APR,反之,使用NIO);

3.BIO/NIO有何不同

无论是BIO,还是NIO,Connector处理请求的大致流程是一样的:
在accept队列中接收连接(当客户端向服务器发送请求时,如果客户端与服务器完成三次握手建立了连接,
则将该连接加入到accept队列中),在连接中获取数据,生产request对象,调用servlet容器处理请求,返回reponse对象。
说明:连接是TCP层面的(传输层),请求是HTTP层面的(应用层),一个TCP连接中可能传输多个HTTP请求。

(1)在BIO实现的Connector中,处理请求的主要实体是:JIoEndpoint对象。
JIoEndpoint对象维护了Acceptor和Worker,Acceptor接收socket,然后从Worker线程池中找出空闲的线程处理socket;
如果Worker没有空闲线程,那么Acceptor将阻塞。
Worker是tomcat自带的线程池,如果通过配置了其他线程池,原理与Worker类似。
server.xml:

<!--The connectors can use a shared executor, you can define one or more named thread pools-->
    <!--
    <Executor name="tomcatThreadPool" namePrefix="catalina-exec-"
        maxThreads="150" minSpareThreads="4"/>
    -->

(2)在NIO实现的Connector中,处理请求的主要实体是NIoEndpoint对象。
NIoEndpoint对象除了包含Acceptor和Worker外,还用了Poller。
Acceptor接收socket后,不是之间使用Worker中的线程处理请求,而是先将请求发给Poller,因为Poller是实现NIO的关键!
Acceptor向Poller发送请求是通过队列实现,使用了生产者-消费者模式。
在Poller中维护了一个Selector对象,当Poller从队列获取socket后,注册到Selector中,然后遍历Selector,
找出其中可读的socket,并使用Worker线程处理请求。与BIO类似,Worker也可以被自定义的线程池代替。

通过上述过程可以看出,在NIoEndpoint处理请求的过程中,无论是Acceptor接收socket,还是线程处理请求,
使用的仍然是阻塞方式;
但在“读取socket并交给Worker中的线程”的这个过程中,使用非阻塞的NIO实现,
这是NIO模式与BIO模式的最主要区别。而这个区别,在并发量较大的情形下可以带来Tomcat效率的显著提升。
目前大多数HTTP请求使用的是长连接(HTTP/1.1默认keep-alive为true),
而长连接意味着,一个TCP的socket在当前请求结束后,如果没有新的请求到来,socket不会立马释放
,而是等timeout后再释放。如果使用BIO,“读取socket并交给Worker中的线程”这个过程是阻塞的,
也就意味着在socket等待下一个请求或等待释放的过程中,处理这个socket的工作线程会一直被占用,
无法释放;因此Tomcat可以同时处理的socket数目不能超过最大线程数,性能受到了极大限制。
而使用NIO,“读取socket并交给Worker中的线程”这个过程是非阻塞的,当socket在等待下一个请求或等待释放时,
并不会占用工作线程,因此Tomcat可以同时处理的socket数目远大于最大线程数,并发性能大大提高。

三.acceptCount、maxConnections、maxThreads

1.acceptCount
accept队列的长度,当accept队列中连接的个数达到acceptCount时,队列满,之后进来的请求一律被拒绝。默认值是100。

2.maxConnections
Tomcat在任意时刻接收和处理的最大连接数。当Tomcat接收的连接数达到maxConnections时,
Acceptor线程不会读取accept队列中的连接;这时accept队列中的线程会一直阻塞着,
直到Tomcat接收的连接数小于maxConnections。如果设置为-1,则连接数不受限制。
默认值与连接器使用的协议有关:
NIO的默认值是10000;
APR/native的默认值是8192;
在windows下,APR/native的maxConnections值会自动调整为设置值以下最大的1024的整数倍;
如设置为2000,则最大值实际是1024。
BIO的默认值为maxThreads(如果配置了Executor,则默认值是Executor的maxThreads)。

3.maxThreads
处理请求的线程的最大数量。默认值是200(Tomcat7和8都是的)。
如果该Connector绑定了Executor,这个值会被忽略,因为该Connector将使用绑定的Executor,而不是内置的线程池来执行任务。
maxThreads规定的是最大的线程数目,并不是实际running的CPU数量;实际上,maxThreads的大小比CPU核心数量要大得多。
这是因为,处理请求的线程真正用于计算的时间可能很少,大多数时间可能在阻塞,如等待数据库返回数据、等待硬盘读写数据等。
因此,在某一时刻,只有少数的线程真正的在使用物理CPU,大多数线程都在等待;因此线程数远大于物理核心数才是合理的。
Tomcat通过使用比CPU核心数量多得多的线程数,可以使CPU忙碌起来,大大提高CPU的利用率。

4.参数设置
(1)maxThreads的设置既与应用的特点有关,也与服务器的CPU核心数量有关。
maxThreads数量应该远大于CPU核心数量;而且CPU核心数越大,maxThreads应该越大;
应用中CPU越不密集(IO越密集),maxThreads应该越大,以便能够充分利用CPU。
当然,maxThreads的值并不是越大越好,如果maxThreads过大,那么CPU会花费大量的时间用于线程的切换,整体效率会降低。
(2)maxConnections的设置与Tomcat的运行模式有关,如果tomcat使用的是BIO,
那么maxConnections的值应该与maxThreads一致;如果tomcat使用的是NIO,
那么类似于Tomcat的默认值,maxConnections值应该远大于maxThreads。
(3)虽然tomcat同时可以处理的连接数目是maxConnections,
但服务器中可以同时接收的连接数为acceptCount 。maxConnections+acceptCount的设置,
与应用在连接过高情况下希望做出什么反应有关系。
如果设置过大,后面进入的请求等待时间会很长;如果设置过小,后面进入的请求立马返回connection refused。

四:线程池Executor
Executor元素代表Tomcat中的线程池,可以由其他组件共享使用;要使用该线程池,
组件需要通过executor属性指定该线程池。Executor是Service元素的内嵌元素。
一般来说,使用线程池的是Connector组件;为了使Connector能使用线程池,Executor元素应该放在Connector前面。
Executor与Connector的配置举例如下:

<Executor name="tomcatThreadPool" namePrefix ="catalina-exec-" 
maxThreads="150" minSpareThreads="4" />
<Connector executor="tomcatThreadPool" port="8080" protocol="HTTP/1.1" 
connectionTimeout="20000" redirectPort="8443" acceptCount="1000" />

Executor的主要属性包括:
name:该线程池的标记
maxThreads:线程池中最大活跃线程数,默认值200(Tomcat7和8都是)
minSpareThreads:线程池中保持的最小线程数,最小值是25
maxIdleTime:线程空闲的最大时间,当空闲超过该值时关闭线程
(除非线程数小于minSpareThreads),单位是ms,默认值60000(1分钟)
daemon:是否后台线程,默认值true
threadPriority:线程优先级,默认值5
namePrefix:线程名字的前缀,线程池中线程名字为:namePrefix+线程编号

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值