SpringBoot監(jiān)控Tomcat活動線程數(shù)來判斷是否完成請求處理方式
SpringBoot監(jiān)控Tomcat活動線程數(shù)來判斷是否完成請求處理
最近項目中有一個需求,需要判斷應用的請求是否已經(jīng)處理完畢,想了一下,打算通過定時任務定時監(jiān)控Tomcat的活動線程數(shù)來實現(xiàn)。
編碼實現(xiàn)
新建一個SpringBoot工程,添加定時任務定時監(jiān)控,代碼如下:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.web.embedded.tomcat.TomcatWebServer;
import org.springframework.boot.web.servlet.context.ServletWebServerApplicationContext;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class TomcatMonitor {
? ? private Logger logger = LoggerFactory.getLogger(getClass());
? ? @Autowired
? ? ServletWebServerApplicationContext applicationContext;
? ? @Scheduled(cron = "*/1 * * * * ?")
? ? public void execute() {
? ? ? ? TomcatWebServer webServer = (TomcatWebServer) applicationContext.getWebServer();
? ? ? ? String tomcatInfo = webServer.getTomcat().getConnector().getProtocolHandler().getExecutor().toString();
? ? ? ? System.out.println("tomcatInfo: " + tomcatInfo);
? ? }
}測試
新建一個測試請求類,代碼如下:
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.concurrent.TimeUnit;
@RestController
public class TestController {
? ? @RequestMapping("/test")
? ? public String test() {
? ? ? ? try {
? ? ? ? ? ? TimeUnit.SECONDS.sleep(2);
? ? ? ? } catch (InterruptedException e) {
? ? ? ? ? ? e.printStackTrace();
? ? ? ? }
? ? ? ? System.out.println("complete OK");
? ? ? ? return "OK";
? ? }
}啟動項目:
tomcatInfo: org.apache.tomcat.util.threads.ThreadPoolExecutor@49355b92[Running, pool size = 10, active threads = 0, queued tasks = 0, completed tasks = 0]
tomcatInfo: org.apache.tomcat.util.threads.ThreadPoolExecutor@49355b92[Running, pool size = 10, active threads = 0, queued tasks = 0, completed tasks = 0]
訪問測試接口,觀察Tomcat線程池變化:
tomcatInfo: org.apache.tomcat.util.threads.ThreadPoolExecutor@49355b92[Running, pool size = 10, active threads = 1, queued tasks = 0, completed tasks = 0]
complete OK
tomcatInfo: org.apache.tomcat.util.threads.ThreadPoolExecutor@49355b92[Running, pool size = 10, active threads = 0, queued tasks = 0, completed tasks = 2]
可以看到,當有請求處理時,Tomcat的活動線程數(shù)不為0,處理完成以后回歸到0,所以可以通過活動線程數(shù)判斷當前Tomcat是否有請求還在處理,當然,應用到實際業(yè)務中,還有需要完善的地方。
Tomcat線程數(shù)占滿而導致的線上事故
事故表現(xiàn)形式
昨天下午4點多的時候,有同學反映說,某個服務響應時間過長,懷疑是負載均衡的問題。
排查過程
1,查看阿里云負載均衡表現(xiàn)正常,排除。
2,查看單臺服務日志出現(xiàn)OOM,可以斷定是服務出問題了。
分析原因
1,先重啟服務。
2,下載 *.hprof 日志,用MAT分析結(jié)果如下圖:
3,結(jié)果明顯指示tomcat線程池滿了,并指向了其中一個controller的外部請求。
4,結(jié)合代碼分析是請求外部HTTP連接沒有設置超時時間,導致大量請求堆積,tomcat線程池滿了,造成OOM。
解決辦法
1,HTTP請求使用連接池,設置超時響應時間
2,加上監(jiān)控。


總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Springboot通過配置WebMvcConfig處理Cors非同源訪問跨域問題
這篇文章主要介紹了Springboot通過配置WebMvcConfig處理Cors非同源訪問跨域問題,關于Cors跨域的問題,前端有代理和jsonp的常用方式解決這種非同源的訪問拒絕策略2023-04-04
解析分別用遞歸與循環(huán)的方式求斐波那契數(shù)列的實現(xiàn)方法
本篇文章是對分別用遞歸與循環(huán)的方式求斐波那契數(shù)列的方法進行了詳細的分析介紹,需要的朋友參考下2013-06-06
SpringCloud微服務架構(gòu)實戰(zhàn)之微服務治理功能的實現(xiàn)
這篇文章主要介紹了SpringCloud微服務架構(gòu)實戰(zhàn)之微服務治理,這些治理工具主要包括服務的注冊與發(fā)現(xiàn)、負載均衡管理、動態(tài)路由、服務降級和故障轉(zhuǎn)移、鏈路跟蹤、服務監(jiān)控等,需要的朋友可以參考下2022-02-02
解決Eclipse打開.java文件異常,提示用系統(tǒng)工具打開的問題
這篇文章主要介紹了解決Eclipse打開.java文件異常,提示用系統(tǒng)工具打開的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-01-01
關于springboot集成swagger及knife4j的增強問題
這篇文章主要介紹了springboot集成swagger以及knife4j的增強,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03
Mybatis動態(tài)查詢字段及表名的實現(xiàn)
本文主要介紹了Mybatis動態(tài)查詢字段及表名的實現(xiàn),通過靈活運用Mybatis提供的動態(tài)SQL功能,我們可以構(gòu)建更加靈活、高效的查詢語句,具有一定的參考價值,感興趣的小伙伴們可以參考一下2024-01-01

