在日常開(kāi)發(fā)工作中厢绝,為了加快反應(yīng)速度(如方法接口的快速返回剑肯,異步做一些操作)或者充分利用資源(正常程序都會(huì)起多線程,來(lái)充分利用資源)欺旧,都會(huì)需要多線程的處理铸屉。在啟用一個(gè)線程的時(shí)候,盡量的要使用線程池切端,好處很多彻坛,如減少新建線程開(kāi)銷,易于控制線程總數(shù)等。
Java中使用線程池昌屉,一般會(huì)調(diào)用的Executors工廠方法钙蒙,常用的有Executors.newSingleThreadScheduledExecutor(),Executors.newFixedThreadPool(10)和Executors.newCachedThreadPool()间驮,當(dāng)然你也可以自己去new一個(gè)ThreadPoolExecutor躬厌。
在這三個(gè)常用的線程池中,上邊是最簡(jiǎn)單的調(diào)用方法竞帽,還可以根據(jù)個(gè)人需要加入ThreadFactory(java的線程生成工廠扛施,可以自己重寫做些命名日志之類)參數(shù)。
下邊主要講下區(qū)別及應(yīng)用場(chǎng)景:
1屹篓。newSingleThreadExecutor會(huì)生成一個(gè)單線程的線程池疙渣,這個(gè)線程永久存在,所有加入這個(gè)線程池的任務(wù)都會(huì)順序的執(zhí)行堆巧,并且能保證線程安全妄荔,因?yàn)槌绦蚴窃趩尉€程中執(zhí)行。
主要應(yīng)用場(chǎng)景:一些不太重要的收尾谍肤,日志等工作可以放到單線程的線程中去執(zhí)行啦租。如,一個(gè)金幣的扣除工作荒揣,你可能需要操作數(shù)據(jù)庫(kù)進(jìn)行金幣加減篷角,并且寫入日志,這兩個(gè)操作中主要的是金幣加減操作系任,他成功了就說(shuō)明方法是調(diào)用成功的内地,而且日志記錄一般情況會(huì)比較慢(數(shù)據(jù)量大一般可能不寫入數(shù)據(jù)庫(kù)),順序執(zhí)行會(huì)拖慢整個(gè)接口赋除,堆積更多請(qǐng)求阱缓,還可能會(huì)對(duì)數(shù)據(jù)庫(kù)造成影響(事務(wù)在開(kāi)啟中),所以日志記錄完全可以扔到單線程的線程中去举农,一條條的處理荆针,也可以認(rèn)為是一個(gè)單消費(fèi)者的生產(chǎn)者消費(fèi)者模式。
2颁糟。newFixedThreadPool會(huì)生成固定線程數(shù)(由你自己指定)的線程池航背。
主要應(yīng)用場(chǎng)景:固定線程數(shù)的線程池比較常見(jiàn),如處理網(wǎng)絡(luò)請(qǐng)求等棱貌。這種主要是為了充分利用系統(tǒng)資源玖媚,如果計(jì)算較多的程序中,線程數(shù)=系統(tǒng)最大并行數(shù)+1(+1為了防止線程掛掉等情況)婚脱,但我們的程序不可能一直在計(jì)算今魔,大部分受限與網(wǎng)絡(luò)硬盤等外部因素勺像,所以線程數(shù)必須加大,來(lái)充分利用系統(tǒng)資源错森。
3吟宦。newCachedThreadPool會(huì)無(wú)限制的生成線程,但如果一個(gè)線程工作完成會(huì)自動(dòng)回收銷毀機(jī)制
主要應(yīng)用場(chǎng)景:需要很極致的速度涩维,因?yàn)閚ewCachedThreadPool不會(huì)等待空閑線程殃姓。但有一定風(fēng)險(xiǎn),如果一個(gè)任務(wù)很慢或者阻塞瓦阐,并且請(qǐng)求很多蜗侈,就容易造成線程泛濫,會(huì)導(dǎo)致整個(gè)系統(tǒng)的假死(無(wú)法接收處理新的請(qǐng)求)睡蟋,所以實(shí)際上個(gè)人不建議使用這個(gè)方法踏幻。
最后,要說(shuō)明下使用newSingleThreadExecutor和newFixedThreadPool的時(shí)候注意任務(wù)間的鎖薄湿,如果任務(wù)之間有依賴鎖(wait/notify),很容易造成死鎖偷卧。