問題來自一個線上GC頻繁的應用,觀察到老年代一直gc下不去導致應用被gc STW卡主假死,檢查代碼發現這樣一段代碼,感覺可疑
代碼如下:
public class WriteEsWork {
public static void write(List<EsIndexInfo> esList, String index, ESClusterEnum cluster, Worker worker) {
execServer.submit(new WriteESRunnable(esList, index, cluster, worker));
}
private static class WriteESRunnable implements Runnable {
private List<EsIndexInfo> esList;
...
}
}
jmap查到WriteESRunnable 這個對象有不少8000多個,一個對象等于一個線程,EsIndexInfo這個對象也很多。
問題:WriteESRunnable 是一個靜態內部類,這個類只會在靜態方法write被調用的時候 new對象到線程池,那么當這個線程執行完成后,WriteESRunnable 對象會被釋放嗎?還是因為他是內部靜態類會一直保留引用?如果不釋放就說明確實是因為這個問題導致WriteESRunnable 和EsIndexInfo對象堆積太久。
如果釋放的話 那就是另一種可能 線程再線程池等待隊列堆積的太多了。
還請朋友們幫忙分析!謝謝
1 回答

倚天杖
TA貢獻1828條經驗 獲得超3個贊
根據你的代碼片段,只有線程池持有對[new WriteESRunnable]的引用,所以第二情況可能性比較大。
每個任務的處理太長, 任務隊列沒有限制導致過長,然后就發生堆積情況了。
添加回答
舉報
0/150
提交
取消