打印Java程序的線程棧信息方式
打印Java程序的線程棧信息
jstack可以得知當(dāng)前線程的運(yùn)行情況
安裝jstack等命令集,jstack是開發(fā)版本jdk的一部分,不是開發(fā)版的有可能找不到
yum install -y java-1.8.0-openjdk-devel
查看要打印堆棧的java進(jìn)程ID
jps -l
打印堆棧
sudo -u admin jstack pid > jstack.txt
特別要注意的是jstack需要使用與進(jìn)程一致的用戶才能正確導(dǎo)出堆棧,否則會(huì)報(bào)錯(cuò)如下
Unable to open socket file: target process not responding or HotSpot VM not loaded
線程池異常堆棧的坑
import java.util.concurrent.*;
public class DivTask implements Runnable{
int a,b;
public DivTask(int a, int b) {
this.a = a;
this.b = b;
}
@Override
public void run() {
double re = a/b;
System.out.println(re);
}
public static void main(String[] args) throws InterruptedException, ExecutionException {
// ThreadPoolExecutor executor = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 0L, TimeUnit.SECONDS
// , new SynchronousQueue<>());
TraceThreadPoolExecutor executor = new TraceThreadPoolExecutor(0, Integer.MAX_VALUE, 0L, TimeUnit.SECONDS
, new SynchronousQueue<>()); //擴(kuò)展TraceThreadPoolExecutor
for (int i = 0; i < 5; i++) {
// executor.submit(new DivTask(100,i));
//改進(jìn)方式一:
//Future re = executor.submit(new DivTask(100, i));
//re.get();
//改進(jìn)方式二:
executor.execute(new DivTask(100,i));
}
//100.0
//25.0
//33.0
//50.0
//其中100/0的異常結(jié)果沒打印
//線程池很有可能"吃掉程序拋出的異常
//改進(jìn)方式一:
//Exception in thread "main" java.util.concurrent.ExecutionException: java.lang.ArithmeticException: / by zero
// at java.util.concurrent.FutureTask.report(FutureTask.java:122)
// at java.util.concurrent.FutureTask.get(FutureTask.java:192)
//。。。
//改進(jìn)方式二:
//Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero
// at com.Test.DivTask.run(DivTask.java:15)
// at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
// at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
// at java.lang.Thread.run(Thread.java:748)
//100.0
//33.0
//25.0
//50.0
//擴(kuò)展TraceThreadPoolExecutor
//java.lang.Exception: Client stack trace
// at com.Test.TraceThreadPoolExecutor.clientTrace(TraceThreadPoolExecutor.java:20)
// at com.Test.TraceThreadPoolExecutor.execute(TraceThreadPoolExecutor.java:12)
// at com.Test.DivTask.main(DivTask.java:29)
//Exception in thread "pool-1-thread-1" java.lang.ArithmeticException: / by zero
// at com.Test.DivTask.run(DivTask.java:15)
// at com.Test.TraceThreadPoolExecutor.lambda$wrap$0(TraceThreadPoolExecutor.java:25)
// at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
// at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
// at java.lang.Thread.run(Thread.java:748)
//100.0
//25.0
//33.0
//50.0
}
}
import java.util.concurrent.*;
/**
* 擴(kuò)展TraceThreadPoolExecutor,讓它在調(diào)度任務(wù)前先保存一下提交任務(wù)線程的堆棧信息
*/
public class TraceThreadPoolExecutor extends ThreadPoolExecutor {
public TraceThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue) {
super(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue);
}
@Override
public void execute(Runnable task) {
super.execute(wrap(task,clientTrace(),Thread.currentThread().getName()));
}
@Override
public Future<?> submit(Runnable task) {
return super.submit(wrap(task,clientTrace(),Thread.currentThread().getName()));
}
private Exception clientTrace(){
return new Exception("Client stack trace");
}
private Runnable wrap(final Runnable task,final Exception clientTrace,String clientThreadName){
return () -> {
try {
task.run();
} catch (Exception e) {
clientTrace.printStackTrace();
throw e;
}
};
}
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
使用Spring?Boot快速構(gòu)建一個(gè)簡(jiǎn)單的文件處理工具
在現(xiàn)代Web應(yīng)用中,文件上傳與處理是常見的需求,本文將通過一個(gè)實(shí)際案例,詳細(xì)介紹如何使用Spring?Boot構(gòu)建一個(gè)文件處理工具,感興趣的小伙伴可以參考一下2025-06-06
JavaFX桌面應(yīng)用未響應(yīng)問題解決方案
這篇文章主要介紹了JavaFX桌面應(yīng)用未響應(yīng)問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07
Spring?MVC??接受請(qǐng)求參數(shù)的方法
了解HTTP請(qǐng)求的GET和POST方法中如何攜帶參數(shù),以及SpringMVC中如何接收這些參數(shù),GET方法通過URL傳遞參數(shù),而POST方法通常在請(qǐng)求體中傳遞,SpringMVC使用注解如@RequestParam和@RequestBody來綁定參數(shù)到控制器方法2024-09-09
Springboot引入hibernate配置自動(dòng)建表并進(jìn)行增刪改查操作
這篇文章主要介紹了Springboot引入hibernate配置自動(dòng)建表并進(jìn)行增刪改查,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-09-09
spring boot自定義配置時(shí)在yml文件輸入有提示問題及解決方案
自定義一個(gè)配置類,然后在yml文件具體配置值時(shí),一般不會(huì)有提示,今天小編給大家分享spring boot自定義配置時(shí)在yml文件輸入有提示問題,感興趣的朋友一起看看吧2023-10-10
PowerJob的TransportServiceAware工作流程源碼解讀
這篇文章主要介紹了PowerJob的TransportServiceAware工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01
Spring容器擴(kuò)展機(jī)制的實(shí)現(xiàn)原理
這篇文章主要介紹了Spring容器擴(kuò)展機(jī)制的實(shí)現(xiàn)原理,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-11-11
Java實(shí)現(xiàn)定時(shí)任務(wù)的示例代碼
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)定時(shí)任務(wù)的相關(guān)知識(shí),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-11-11
Mybatis中l(wèi)ike搭配concat的寫法詳解
這篇文章主要介紹了Mybatis中l(wèi)ike搭配concat的寫法詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01

