線程池做什么
網絡請求通常有兩種形式:
第一種,請求不是很頻繁,而且每次連接后會保持相當一段時間來讀數據或者寫數據,最后斷開,如文件下載,網絡流媒體等。
另一種形式是請求頻繁,但是連接上以后讀/寫很少量的數據就斷開連接。考慮到服務的并發問題,如果每個請求來到以后服務都為它啟動一個線程,那么這對服務的資源可能會造成很大的浪費,特別是第二種情況。
因為通常情況下,創建線程是需要一定的耗時的,設這個時間為t1,而連接后讀/寫服務的時間為t2,當t1>>t2時,我們就應當考慮一種策略或者機制來控制,使得服務對于第二種請求方式也能在較低的功耗下完成。
通常,我們可以用線程池來解決這個問題,首先,在服務啟動的時候,我們可以啟動好幾個線程,并用一個容器(如線程池)來管理這些線程。
當請求到來時,可以從池中取一個線程出來,執行任務(通常是對請求的響應),當任務結束后,再將這個線程放入池中備用;
如果請求到來而池中沒有空閑的線程,該請求需要排隊等候。最后,當服務關閉時銷毀該池即可。
然而最近在開發中用到了java的線程池,然后就很疑惑這個線程池到底要不要手動關閉,感覺是要關閉的,但是沒人強調線程池用完要關閉。so今天來試驗下到底線程池用完要不要關閉。
直接上實驗代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
public static void main(string[] args) throws exception { //用于獲取到本java進程,進而獲取總線程數 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( "當前線程總數為:" +bean.getthreadcount()); }); } } thread.sleep( 10000 ); system.out.println( "線程總數為 = " + bean.getthreadcount()); } |
簡單來說就是在一個 for 循環中創建線程池,然后執行一個打印任務(不執行任務線程不會真正創建),打印出當前 java 進程的總線程數,下面是打印部分結果:
線程
可以看到在創建到 15 萬個線程是爆內存,內存占用百分百后 java 應用崩潰。說明線程未被回收。
ps:內存占用百分百后,部分應用開始出現異常,界面花屏,閃屏,不能正常繪制gui,不知道為啥,即使后面內存占用降下來也一樣,只能重啟應用。
結論
使用完線程池一定記得回收,否則跑著跑著就內存爆炸崩潰。回收函數如下:
1
2
3
4
|
//執行此函數后線程池不再接收新任務,并等待所有任務執行完畢后銷毀線程。此函數不會等待銷毀完畢 executor.shutdown(); //立即結束所有線程,不管是否正在運行,返回未執行完畢的任務列表 executor.shutdownnow(); |
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。
原文鏈接:https://www.cnblogs.com/wuyoucao/p/10247012.html