Java線程池使用與原理詳解
線程池是什么?
我們可以利用java很容易創(chuàng)建一個新線程,同時操作系統(tǒng)創(chuàng)建一個線程也是一筆不小的開銷。所以基于線程的復(fù)用,就提出了線程池的概念,我們使用線程池創(chuàng)建出若干個線程,執(zhí)行完一個任務(wù)后,該線程會存在一段時間(用戶可以設(shè)定空閑線程的存活時間,后面會介紹),等到新任務(wù)來的時候就直接復(fù)用這個空閑線程,這樣就省去了創(chuàng)建、銷毀線程損耗。當(dāng)然空閑線程也會是一種資源的浪費(所有才有空閑線程存活時間的限制),但總比頻繁的創(chuàng)建銷毀線程好太多。
下面是我的測試代碼
/* * @TODO 線程池測試 */ @Test public void threadPool(){ /*java提供的統(tǒng)計線程運行數(shù),一開始設(shè)置其值為50000,每一個線程任務(wù)執(zhí)行完 * 調(diào)用CountDownLatch#coutDown()方法(其實就是自減1) * 當(dāng)所有的線程都執(zhí)行完其值就為0 */ CountDownLatch count = new CountDownLatch(50000); long start = System.currentTimeMillis(); Executor pool = Executors.newFixedThreadPool(10);//開啟線程池最多會創(chuàng)建10個線程 for(int i=0;i<50000;i++){ pool.execute(new Runnable() { @Override public void run() { System.out.println("hello"); count.countDown(); } }); } while(count.getCount()!=0){//堵塞等待5w個線程運行完畢 } long end = System.currentTimeMillis(); System.out.println("50個線程都執(zhí)行完了,共用時:"+(end-start)+"ms"); } /** *@TODO 手動創(chuàng)建線程測試 */ @Test public void thread(){ CountDownLatch count = new CountDownLatch(50000); long start = System.currentTimeMillis(); for(int i=0;i<50000;i++){ Thread thread = new Thread(new Runnable() { @Override public void run() { System.out.println("hello"); count.countDown(); } }); thread.start(); } while(count.getCount()!=0){//堵塞等待5w個線程運行完畢 } long end = System.currentTimeMillis(); System.out.println("50000個線程都執(zhí)行完了,共用時:"+(end-start)+"ms"); }
使用線程池5w線程運行完大約為400ms,不使用線程池運行大約為4350ms左右,其效率可見一斑(讀者可以自行測試,不過由于電腦配置不一樣,跑出來的數(shù)據(jù)會有差別,但使用線程池絕對是比創(chuàng)建線程要快的)。
java如何使用線程池?
上面的測試代碼中已經(jīng)使用了線程池,下面正式介紹一下。
java所有的線程池最頂層是一個Executor接口,其只有一個execute方法,用于執(zhí)行所有的任務(wù),java又提供了ExecutorService接口繼承自Executor并且擴(kuò)充了一下方法,在往下就是AbstractExecutorService這個抽象類,其實現(xiàn)了ExecutorService,最后就是ThreadPoolExecutor其繼承自上面的抽象類,我們常使用的java線程池就是創(chuàng)建的這個類的實例。
而上面我們使用Executors是一個工具類,它就是一個語法糖,為我們把各種不同的業(yè)務(wù)的線程池參數(shù)進(jìn)行封裝,進(jìn)行new操作。
public static ExecutorService newFixedThreadPool(int nThreads) { return new ThreadPoolExecutor(nThreads, nThreads, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>()); }
上面就是Executors.newFixedThreadPool(10)的源碼。
下面重點來了,說一說ThreadPoolExecutor構(gòu)造方法各參數(shù)的意思。
public ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler)
上面這個構(gòu)造方法是最全的。
下面我們根據(jù)源碼來解釋部分參數(shù)意思,這樣更有說服力。
下面是ThreadPoolExecutor#execute方法,就是我們上面接口調(diào)用的execute實際執(zhí)行者。
public void execute(Runnable command) { if (command == null) throw new NullPointerException(); int c = ctl.get(); if (workerCountOf(c) < corePoolSize) { if (addWorker(command, true)) return; c = ctl.get(); } if (isRunning(c) && workQueue.offer(command)) { int recheck = ctl.get(); if (! isRunning(recheck) && remove(command)) reject(command); else if (workerCountOf(recheck) == 0) addWorker(null, false); } else if (!addWorker(command, false)) reject(command); }
ctl是一個AtomicInteger實例,是一個提供了原子語句的CAS操作的類,它用來記錄線程池中當(dāng)前運行的線程數(shù)量加上-2^29,workCountOf方法就取得其絕對值(可以去看源碼如何實現(xiàn)),當(dāng)其小于corePoolSize時,會調(diào)用addWorker方法(是用來創(chuàng)建一個新Workder,Workder會創(chuàng)建一個Thread,所以就是創(chuàng)建線程的方法),addWorkd創(chuàng)建線程過程中會跟corePoolSize或者maxnumPoolSize的值比較(當(dāng)傳入true會根corePoolSize比較,false會根據(jù)maxnumPoolSize比較,大于等于其值會創(chuàng)建失?。???梢娙绾萎?dāng)前運行中的線程數(shù)量小于corePoolSize就是創(chuàng)建并且也會創(chuàng)建成功(
只簡單的討論線程池Running狀態(tài)下)。
如果當(dāng)運行中線程數(shù)大于等于corePoolSize時,進(jìn)入第二個if,isRunning是跟SHUTDOWN(其值=0)比較,之前說過c等于當(dāng)前運行的線程數(shù)量加上-2^29,如果當(dāng)前當(dāng)前運行的線程數(shù)據(jù)達(dá)到2^29時其值就=0,isRunning返回false,else中在執(zhí)行addWorkd也會返回false(addWorkd也對其進(jìn)行了檢驗),所以這表示線程池最多能支持2^29個線程同時運行(足夠用了)。
workQueue.offer(command)就是將runnable加入等待隊列,加入等待隊列后runWorker方法會從隊列中獲取任務(wù)執(zhí)行的。如果當(dāng)前隊列采用的是有界隊列(ArrayBlockingQueue)當(dāng)隊列滿了offer就會返回false,這是就進(jìn)入else if,看!這里傳入了false,說明這里要跟maxnumPoolSize比較了,如果這里運行的線程數(shù)大于等于maxnumPoolSize,那么這個線程任務(wù)就要被線程池拒絕了,執(zhí)行reject(command),拒絕方法中使用了我們ThreadPoolExecutor構(gòu)造方法中的RejectedExecutionHandler(拒絕策略),后面再詳細(xì)解釋。
經(jīng)過上面的結(jié)合源碼的介紹,下面對們ThreadPoolExecutor的參數(shù)介紹就好理解了。
線程池中線程創(chuàng)建和拒絕策略
corePoolSize,maxnumPoolSize,BlockingQueue這三個要一塊說
當(dāng)線程池運行的線程小于corePoolSize時,來一個新線程任務(wù)總是會新建一個線程來執(zhí)行;當(dāng)大于corePoolSize就會把任務(wù)加入到等待隊列blockingQueue中,如果你傳入的BlockingQueue是一個無界隊列(LinkedBlockingQueue)這是隊列可以存放“無窮多”的任務(wù),所有總是會加入隊列成功,跟maxnumPoolSize就沒關(guān)系了,這也表示線程池中線程數(shù)最多為corePoolSize個;但是如果你傳入的是有界隊列(ArrayBlockingQueue,SynchronousQueue),當(dāng)隊列滿時,并且線程數(shù)小于maxmunPoolSize就是創(chuàng)建新的線程直至線程數(shù)大于maxnumPoolSize;如果當(dāng)線程數(shù)量大于maxnumPoolSize時,在加入任務(wù)就會被線程池拒絕。
RejectedExecutionHandler拒絕策略java給實現(xiàn)了4個AbortPolicy,CallerRunsPolicy,DiscardOldestPolicy,DiscardPolicy用戶也可以自己實現(xiàn)該接口實現(xiàn)自己的拒絕策略;第一個就是直接拋出異常,我們可以進(jìn)行trycatch處理;第二個就是該新任務(wù)直接運行;第三個是取消隊列中最老的;第四個是取消當(dāng)前任務(wù)。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Java中LinkedList詳解和使用示例_動力節(jié)點Java學(xué)院整理
LinkedList 是一個繼承于AbstractSequentialList的雙向鏈表。它也可以被當(dāng)作堆棧、隊列或雙端隊列進(jìn)行操作。接下來通過示例代碼給大家詳細(xì)介紹java中l(wèi)inkedlist的使用,需要的朋友參考下吧2017-05-05Java經(jīng)典面試題匯總:網(wǎng)絡(luò)編程
本篇總結(jié)的是Java 網(wǎng)絡(luò)編程相關(guān)的面試題,后續(xù)會持續(xù)更新,希望我的分享可以幫助到正在備戰(zhàn)面試的實習(xí)生或者已經(jīng)工作的同行,如果發(fā)現(xiàn)錯誤還望大家多多包涵,不吝賜教,謝謝2021-07-07springcloud整合到項目中無法啟動報錯Failed to start bean&n
這篇文章主要介紹了springcloud整合到項目中無法啟動報錯Failed to start bean 'eurekaAutoServiceRegistration'問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01idea啟動springboot報錯: 找不到或無法加載主類問題
這篇文章主要介紹了idea啟動springboot報錯: 找不到或無法加載主類問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12Java?SE循環(huán)一些基本練習(xí)題總結(jié)
循環(huán)語句可以在滿足循環(huán)條件的情況下,反復(fù)執(zhí)行某一段代碼,這段被重復(fù)執(zhí)行的代碼被稱為循環(huán)體語句,下面這篇文章主要給大家總結(jié)介紹了關(guān)于Java?SE循環(huán)一些基本練習(xí)題,需要的朋友可以參考下2024-03-03