我們的軟件傾向于使用大量的線程池——主要是通過(guò)?j?ava.util.concurrent.ExecutorService?
?實(shí)現(xiàn)(通過(guò)?Executors.new...
?創(chuàng)建)。我們?yōu)楦鞣N異步用例創(chuàng)建這些線程池,它們隨處可見(jiàn)。所有這些執(zhí)行器都有一個(gè)線程工廠。它隱藏在默認(rèn)工廠方法,但您可以提供線程工廠。如果未提供,則在需要線程時(shí)使用默認(rèn)線程工廠。
使用 spring 時(shí),可以使用?<task:executor />
?。在這種情況下,每個(gè)執(zhí)行程序服務(wù)的線程工廠由 spring 提供,它使用執(zhí)行程序 bean 的名稱(用?id="executorName"
?指定)。但是對(duì)于那些不是由 spring 創(chuàng)建的,使用默認(rèn)名稱是沒(méi)有幫助的,并且不允許您按名稱區(qū)分線程。
并且您需要按名稱區(qū)分線程——如果出現(xiàn)性能問(wèn)題,您有多種調(diào)查選項(xiàng):線程轉(zhuǎn)儲(chǔ)和使用 top 命令。在這兩種情況下,了解線程服務(wù)的函數(shù)是有用的,因?yàn)檗D(zhuǎn)儲(chǔ)中的堆棧跟蹤可能并不總是顯示出來(lái)。
我最喜歡的快速調(diào)查工具是?top
?,更準(zhǔn)確地說(shuō),?top -H -p <pid>
?。這顯示了通常的頂部表,但 ?-H
?標(biāo)志意味著應(yīng)該打印所選進(jìn)程的線程。按名稱,您基本上會(huì)獲得占用 CPU 最多且當(dāng)前處于活動(dòng)狀態(tài)的線程。在這些情況下,自定義名稱非常有用。
但是如何設(shè)置名稱呢?通過(guò)在創(chuàng)建每個(gè)執(zhí)行程序時(shí)指定一個(gè)命名的線程工廠。
我使用的方法基于第二個(gè)答案:
public class AsyncUtils {
public static ThreadFactory createNamedThreadFactory(String name) {
return new ThreadFactoryBuilder().setNameFormat(name + "-%d").build();
}
}
通過(guò) spring 集中管理所有 executor 可能是一個(gè)更好的主意,但并不是每個(gè)人都在使用 spring,有時(shí)甚至可能超出 spring bean 的一小部分功能需要一個(gè) executor。因此,這是一個(gè)很好的方法。