網(wǎng)站首頁 編程語言 正文
如何給 tomcat 配置合適的線程池
任務分為 CPU 密集型和 IO 密集型
對于 CPU 密集型的應用來說,需要大量 CPU 計算速度很快,線程池如果過多,則保存和切換上下文開銷過高反而會影響性能,可以適當將線程數(shù)量調(diào)小一些
對于 IO 密集型應用來說常見于普通的業(yè)務系統(tǒng),比如會去查詢 mysql、redis 等然后在內(nèi)存中做簡單的數(shù)據(jù)組裝和邏輯判斷,此時由于有 epoll、dma 等支持,消耗較長時間的線程不會長時間占據(jù) CPU 時間,所以可以適當將線程數(shù)量調(diào)大一些
對于線程數(shù)到底該設置多大,網(wǎng)上有很多的方案
我們在實際情況中基于 wrk 等壓測工具進行壓測,壓測邏輯相對復雜請求量相對較高的接口先配置一個相對保守的值
先預估假設當前接口處理平均耗時 300MS,1S 一個線程平均處理 3 個請求,最大 100 個線程 1S 能處理 300 TPS
粗略估算每個請求會消耗多大的內(nèi)容空間,假如是 2KB,那么每秒會消耗 600KB 以此來估算 YGC 多久會觸發(fā)一次,假設年輕代為 1GB 那么大約 29 分鐘觸發(fā)一次 YGC
然后再看 100 個線程持續(xù)處理 300TPS 的時候 CPU 消耗情況
觀察內(nèi)存幾分鐘 GC 一次,或者 CPU 使用率穩(wěn)定在 50% 左右就好,假設此時線程池 70 個線程都在運作
那么監(jiān)控系統(tǒng)采集到線程池線程使用率達到了 80% 就開始擴容,因為擴容拉起新的服務器到提供服務可能也需要1-2分鐘的時間,所以需要預留服務器資源能夠處理彈性擴容期間的流量
實際場景中是相對比較復雜的,前期最好設置一個相對保守的擴容閾值,如果發(fā)現(xiàn)在達到這個閾值前服務器已經(jīng)頂不住了就可以實時的縮小閾值
同時還可以根據(jù)在達到擴容閾值擴容的時候,觀察線上真實的 CPU 和內(nèi)存使用情況,可以基于 promethues + grafana 來進行監(jiān)控,如果發(fā)現(xiàn)擴容時候 CPU 使用率和內(nèi)存使用率 GC 評率比較低,那么可以再配置中心動態(tài)的調(diào)整線程池的大小
所以說與其尋找一個準確的計算線程池數(shù)量的配置方式,不如提供一個可以動態(tài)調(diào)整的線程池作為 tomcat 的線程池
如何監(jiān)控 tomcat 線程池的工作情況
spring boot 在啟動過程中會調(diào)用 TomcatProtocolHandlerCustomizer 的實現(xiàn)類,此處可以自定化 tomcat 線程池,也可以獲取到 tomcat 線程池
public class MyTomcatProtocolHandlerCustomizer implements TomcatProtocolHandlerCustomizer<ProtocolHandler> {
private final ThreadPoolExecutor tomcatThreadPoolExecutor;
public TomcatProtocolHandlerCustomizer(ThreadPoolExecutor tomcatThreadPoolExecutor) {
this.tomcatThreadPoolExecutor = tomcatThreadPoolExecutor;
}
@Override
public void customize(ProtocolHandler protocolHandler) {
protocolHandler.setExecutor(tomcatThreadPoolExecutor);
}
public ThreadPoolExecutor getThreadPoolExecutor() {
return tomcatThreadPoolExecutor;
}
}
然后將線程池裝入一個容器 bean 中注冊到 promethues 監(jiān)控中,當每秒進行采集的時候通過回調(diào)的方法去獲取線程池的最新指標
promethues 每秒采集一次容器的線程池運行指標、服務器測 CPU 使用率當滿足定義的擴容閾值時就拉起新的 POD
grafana 每秒采集一次 promethues 的數(shù)據(jù)匯聚圖標展示
@Bean
public AllServersThreadPoolCollector allServersThreadPoolCollector(@Qualifier(value = "GrpcThreadPoolExecutor") ThreadPoolExecutor GrpcThreadPoolExecutor,
@Qualifier(value = "jdTomcatThreadPoolExecutor") org.apache.tomcat.util.threads.ThreadPoolExecutor TomcatThreadPoolExecutor,
MeterRegistry registry) {
return new AllServersThreadPoolCollector(GrpcThreadPoolExecutor, jdTomcatThreadPoolExecutor, registry);
}
public void init() {
try {
createGauge(this, "grpc_tomcat_core_pool_size", pool -> this.getCorePoolSize());
createGauge(this, "grpc_tomcat_maximum_pool_size", pool -> this.getMaximumPoolSize());
createGauge(this, "grpc_tomcat_busy_pool_size", pool -> this.getActiveCount());
createGauge(this, "grpc_tomcat_wait_queue_size", pool -> this.getWaitQueueSize());
} catch (Exception e) {
log.error("注冊 all servers 監(jiān)控失敗", e);
}
}
private void createGauge(AllServersThreadPoolCollector weakRef, String metric, ToDoubleFunction<AllServersThreadPoolCollector> measure) {
Gauge.builder(metric, weakRef, measure)
.register(this.registry);
}
public int getWaitQueueSize() {
return grpcThreadPoolExecutor.getQueue().size() + tomcatThreadPoolExecutor.getQueue().size();
}
tomcat 線程池擴縮容
Java 線程池是支持直接修改 corePoolSize、maximumPoolSize 的
- 修改 corePoolSize
(1)數(shù)量小于 maximumPoolSize 拋異常
(2)如果 corePoolSize - 原來的 = delta,delta 大于 0 那么創(chuàng)建等待隊列任務數(shù)量和 delta 個線程來處理等待處理的任務
(3)如果正在運行的線程數(shù)量 > corePoolSize 那么就中斷多余的線程
- 修改 maximumPoolSize
(1)maximumPoolSize 小于 corePoolSize 拋錯
(2)如果運行的線程大于 maximumPoolSize 中斷掉一些空閑的線程
基于這些機制就能在運行期間動態(tài)調(diào)整線程池內(nèi)容
無需擔心會中斷掉正在運行的任務,因為線程池 worker 線程每執(zhí)行一個任務的時候
tomcat 是如何避免原生線程池的缺陷的
原生線程池的工作原理
(1)運行的線程數(shù)小于核心線程,就創(chuàng)建一個 worker 線程去執(zhí)行任務
(2)運行的線程數(shù) >= 核心線程了,將任務全部積壓到隊列中
(3)隊列如果滿了繼續(xù)創(chuàng)建非核心線程 worker 去執(zhí)行任務
假如 tomcat 采用了原生線程池,核心線程為 10 個,最大線程為 100,隊列為 200,并發(fā)來了 100 個請求,那么同時系統(tǒng)只能處理 10 個,剩下 90 個都得放入隊列中讓 10 個核心線程慢慢排隊處理,延時必然非常高
tomcat 如何優(yōu)化線程池,核心在于阻塞隊列的實現(xiàn),因為阻塞隊列滿了才會繼續(xù)創(chuàng)建非核心 worker 線程處理任務
(1)運行的線程數(shù)小于核心線程,就創(chuàng)建一個 worker 線程去執(zhí)行任務
(2)當前已經(jīng)創(chuàng)建的核心+非核心線程數(shù)等于最大線程數(shù),任務壓入隊列
(3)正在處理的請求數(shù)量小于核心+非核心線程數(shù),任務壓入隊列
(4)當前已經(jīng)創(chuàng)建的核心+非核心線程數(shù)小于最大線程數(shù),創(chuàng)建 worker 線程處理請求
總結(jié)就是一句話當高并發(fā)流量過來的時候,會去創(chuàng)建最大線程數(shù)的 worker 去處理請求用以降低尾延遲,超過最大線程后,任務將被壓入隊列中進行處理
原文鏈接:https://juejin.cn/post/7141011630300594207
相關推薦
- 2023-11-14 樹莓派上如何安裝anaconda/miniconda環(huán)境配置
- 2022-09-09 pycharm?sql語句警告的處理_python
- 2022-04-05 關于redis客戶端連接不上
- 2023-02-02 C#實現(xiàn)網(wǎng)絡小程序的步驟詳解_C#教程
- 2022-06-13 Openstack各組件邏輯關系及運行流程解析_OpenStack
- 2022-12-15 Python?OpenCV中cv2.minAreaRect實例解析_python
- 2022-09-13 Android?Studio實現(xiàn)智能聊天_Android
- 2022-02-27 微信小程序 - 子組件觸發(fā)父組件函數(shù)(triggerEvent)
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支