溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

java線程池使用后到底要關(guān)閉嗎

發(fā)布時(shí)間:2020-08-19 13:04:14 來源:腳本之家 閱讀:396 作者:煩囂的人 欄目:編程語言

線程池做什么

網(wǎng)絡(luò)請(qǐng)求通常有兩種形式:

第一種,請(qǐng)求不是很頻繁,而且每次連接后會(huì)保持相當(dāng)一段時(shí)間來讀數(shù)據(jù)或者寫數(shù)據(jù),最后斷開,如文件下載,網(wǎng)絡(luò)流媒體等。

另一種形式是請(qǐng)求頻繁,但是連接上以后讀/寫很少量的數(shù)據(jù)就斷開連接??紤]到服務(wù)的并發(fā)問題,如果每個(gè)請(qǐng)求來到以后服務(wù)都為它啟動(dòng)一個(gè)線程,那么這對(duì)服務(wù)的資源可能會(huì)造成很大的浪費(fèi),特別是第二種情況。

因?yàn)橥ǔG闆r下,創(chuàng)建線程是需要一定的耗時(shí)的,設(shè)這個(gè)時(shí)間為T1,而連接后讀/寫服務(wù)的時(shí)間為T2,當(dāng)T1>>T2時(shí),我們就應(yīng)當(dāng)考慮一種策略或者機(jī)制來控制,使得服務(wù)對(duì)于第二種請(qǐng)求方式也能在較低的功耗下完成。

通常,我們可以用線程池來解決這個(gè)問題,首先,在服務(wù)啟動(dòng)的時(shí)候,我們可以啟動(dòng)好幾個(gè)線程,并用一個(gè)容器(如線程池)來管理這些線程。

當(dāng)請(qǐng)求到來時(shí),可以從池中取一個(gè)線程出來,執(zhí)行任務(wù)(通常是對(duì)請(qǐng)求的響應(yīng)),當(dāng)任務(wù)結(jié)束后,再將這個(gè)線程放入池中備用;

如果請(qǐng)求到來而池中沒有空閑的線程,該請(qǐng)求需要排隊(duì)等候。最后,當(dāng)服務(wù)關(guān)閉時(shí)銷毀該池即可。

然而最近在開發(fā)中用到了java的線程池,然后就很疑惑這個(gè)線程池到底要不要手動(dòng)關(guān)閉,感覺是要關(guān)閉的,但是沒人強(qiáng)調(diào)線程池用完要關(guān)閉。so今天來試驗(yàn)下到底線程池用完要不要關(guān)閉。

直接上實(shí)驗(yàn)代碼

public static void main(String[] args) throws Exception {
  //用于獲取到本java進(jìn)程,進(jìn)而獲取總線程數(shù)
 RuntimeMXBean runtimeBean = ManagementFactory.getRuntimeMXBean();
 String jvmName = runtimeBean.getName();
 System.out.println("JVM Name = " + jvmName);
 long pid = Long.valueOf(jvmName.split("@")[0]);
 System.out.println("JVM PID = " + pid);
 ThreadMXBean bean = ManagementFactory.getThreadMXBean();
 int n = 30000;
 for (int i = 0; i < n; i++) {
  ThreadPoolExecutor executor = new ThreadPoolExecutor(10,20,1000,TimeUnit.SECONDS,new LinkedBlockingDeque<>());
  for(int j=0;j<10;j++){
   executor.execute(()->{
    System.out.println("當(dāng)前線程總數(shù)為:"+bean.getThreadCount());
   });
  }
 }
 Thread.sleep(10000);
 System.out.println("線程總數(shù)為 = " + bean.getThreadCount());
}

簡(jiǎn)單來說就是在一個(gè) for 循環(huán)中創(chuàng)建線程池,然后執(zhí)行一個(gè)打印任務(wù)(不執(zhí)行任務(wù)線程不會(huì)真正創(chuàng)建),打印出當(dāng)前 java 進(jìn)程的總線程數(shù),下面是打印部分結(jié)果:

java線程池使用后到底要關(guān)閉嗎

線程

可以看到在創(chuàng)建到 15 萬個(gè)線程是爆內(nèi)存,內(nèi)存占用百分百后 java 應(yīng)用崩潰。說明線程未被回收。

PS:內(nèi)存占用百分百后,部分應(yīng)用開始出現(xiàn)異常,界面花屏,閃屏,不能正常繪制gui,不知道為啥,即使后面內(nèi)存占用降下來也一樣,只能重啟應(yīng)用。

結(jié)論

使用完線程池一定記得回收,否則跑著跑著就內(nèi)存爆炸崩潰?;厥蘸瘮?shù)如下:

//執(zhí)行此函數(shù)后線程池不再接收新任務(wù),并等待所有任務(wù)執(zhí)行完畢后銷毀線程。此函數(shù)不會(huì)等待銷毀完畢
executor.shutdown();
//立即結(jié)束所有線程,不管是否正在運(yùn)行,返回未執(zhí)行完畢的任務(wù)列表
executor.shutdownNow();

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)億速云的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI