理解java多線程中ExecutorService使用
java.util.concurrent包里提供了關(guān)于多線程操作的類,平常用的比較多的是ExecutorService及其實現(xiàn)類(如ThreadPoolExecutor等),Executor,Executors,F(xiàn)uture,Callable等
1. ExecutorService(繼承自Executor)接口:提供了一些異步的多線程操作方法,如execute(), submit(), shutdown(), shutdownNow()等
2. Executor接口:執(zhí)行提交的任務(wù)(線程),只有一個方法 execute(Runnable a)
2. Executors類: 提供了一些工廠方法和一些公共方法來操作Executor子類和ThreadFactory等,如newXXX(),xxxThreadFactory()等
3. Futrue接口:代表了線程執(zhí)行結(jié)果,提供了獲取線程執(zhí)行結(jié)果和取消線程的方法,如get(),cancle()等
4. Callable接口:JDK1.5提供的有返回值的線程執(zhí)行新接口
對ExecutorService和Future的理解做簡單記錄
代碼:
public class Main {
private static int count = 0;
public static void main(String[] args){
List<Future> resultList = new LinkedList<>();
/**
* Executors.newCachedThreadPool() 創(chuàng)建一個線程緩存池,若60s中線程沒有被使用,則會停止線程并從緩存池中移除
* Executors.newScheduledThreadPool() 創(chuàng)建一個固定容量的線程池,里邊的線程按照設(shè)定的調(diào)度時間執(zhí)行
* Executors.newFixedThreadPool() 擁有固定容量的線程緩存池
* Executors.newSingleThreadExecutor() 容量為一的線程緩存池,只會有一個線程
*/
ExecutorService executorService = Executors.newCachedThreadPool();
for(int i=0; i<10; i++){
Future future = executorService.submit(new Callable<String>() {
@Override
public String call() {
try {
System.out.println(Thread.currentThread().getName());
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
int count = Main.count;
System.out.println(Thread.currentThread().getName() + "..start Main count:..." + count);
Main.count = ++count;
System.out.println(Thread.currentThread().getName() + "..end Main count:..." + Main.count);
return Thread.currentThread().getName();
}
});
resultList.add(future);
}
executorService.shutdown();
for(Future future: resultList){
try {
System.out.println(future.get() + "..is over...");
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
System.out.println("main thread end...");
}
}
輸出:
pool-1-thread-1 pool-1-thread-2 pool-1-thread-3 pool-1-thread-4 pool-1-thread-5 pool-1-thread-6 pool-1-thread-7 pool-1-thread-8 pool-1-thread-9 pool-1-thread-10 pool-1-thread-1..start Main count:...0 pool-1-thread-2..start Main count:...0 pool-1-thread-3..start Main count:...1 pool-1-thread-2..end Main count:...1 pool-1-thread-1..end Main count:...1 pool-1-thread-3..end Main count:...2 pool-1-thread-1..is over... pool-1-thread-2..is over... pool-1-thread-4..start Main count:...2 pool-1-thread-3..is over... pool-1-thread-4..end Main count:...3 pool-1-thread-4..is over... pool-1-thread-5..start Main count:...3 pool-1-thread-5..end Main count:...4 pool-1-thread-5..is over... pool-1-thread-6..start Main count:...4 pool-1-thread-6..end Main count:...5 pool-1-thread-6..is over... pool-1-thread-7..start Main count:...5 pool-1-thread-7..end Main count:...6 pool-1-thread-7..is over... pool-1-thread-8..start Main count:...6 pool-1-thread-8..end Main count:...7 pool-1-thread-8..is over... pool-1-thread-9..start Main count:...7 pool-1-thread-9..end Main count:...8 pool-1-thread-9..is over... pool-1-thread-10..start Main count:...8 pool-1-thread-10..end Main count:...9 pool-1-thread-10..is over... main thread end... //主線程在所有線程執(zhí)行完成后結(jié)束
控制臺在等待5秒后打印出上邊的輸出結(jié)果,原因是所有的線程啟動的時候是一個并發(fā)操作,都會去等待5秒,所以整體看來只等了5秒,這是一個并發(fā)操作
總結(jié):
1. ExecutorService提供的execute()方法和submit()方法的區(qū)別:
a. execute()方法只接受Runnable類型的實例,所以不能拿到返回值,也不能動態(tài)獲取線程執(zhí)行的情況
b. submit()方法接受Runnable和Callable實例,會返回Future實例,F(xiàn)uture實例的get()方法可以獲取線程執(zhí)行返回值,并能拋出線程執(zhí)行異常。所以如果要獲取線程執(zhí)行返回的結(jié)果,并能處理線程執(zhí)行時可能出現(xiàn)的異常,或者想中途取消線程執(zhí)行時可以使用submit()方法
2. 通過輸出可以看到main方法(主線程)在所有線程執(zhí)行完成后結(jié)束,原因:
a. 通過submit()方法獲取Future實例,并通過Future實例的get()方法獲取線程返回結(jié)果,而Future實例的get()方法會等待線程執(zhí)行完畢才會返回,所以main方法會等待所有子線程結(jié)束才會結(jié)束
b. 若去掉上邊紅色標(biāo)注的for循環(huán),則main方法(主線程)會提前結(jié)束,而不會等待所有子線程結(jié)束
補充:
1. 多個線程并發(fā)執(zhí)行時,若其中某一個線程出現(xiàn)了異常并且沒有被處理,則該線程會自動停止執(zhí)行,但其他線程還是會正常執(zhí)行,這就是為什么tomcat請求出現(xiàn)異常時,tomcat還可以繼續(xù)提供服務(wù)的原因。
2. tomcat提供了線程池和等待池,每一個請求過來都會重新啟動一個新的線程處理該請求,若線程池中線程用完,再來請求的時候就會放到等待池中等待,當(dāng)其中有線程釋放回線程池中時,就會為等待池中的請求分配線程處理請求。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- java多線程開發(fā)ScheduledExecutorService簡化方式
- java多線程CountDownLatch與線程池ThreadPoolExecutor/ExecutorService案例
- Java使用ExecutorService來停止線程服務(wù)
- 在spring boot中使用java線程池ExecutorService的講解
- Java ExecutorService四種線程池使用詳解
- 詳解JDK中ExecutorService與Callable和Future對線程的支持
- 詳解Java利用ExecutorService實現(xiàn)同步執(zhí)行大量線程
- Java 線程池ExecutorService詳解及實例代碼
- java多線程并發(fā)executorservice(任務(wù)調(diào)度)類
- ExecutorService實現(xiàn)獲取線程返回值
相關(guān)文章
關(guān)于SpringBoot大文件RestTemplate下載解決方案
這篇文章主要介紹了SpringBoot大文件RestTemplate下載解決方案,最近結(jié)合網(wǎng)上案例及自己總結(jié),寫了一個分片下載tuling/fileServer項目,需要的朋友可以參考下2021-10-10
Java詳解如何將excel數(shù)據(jù)轉(zhuǎn)為樹形
在平常的辦公工作中,excel數(shù)據(jù)的操作是最常見的需求,今天就來看一下通過Java如何來實現(xiàn)將excel數(shù)據(jù)轉(zhuǎn)為樹形,感興趣的朋友可以了解下2022-08-08
Java線程池的幾種實現(xiàn)方法和區(qū)別介紹實例詳解
本篇文章主要介紹了Java線程池的幾種實現(xiàn)方法和區(qū)別,需要的朋友可以參考2017-04-04
Java Management Extensions管理擴展原理解析
這篇文章主要介紹了Java Management Extensions管理擴展原理解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-04-04
Java中對AtomicInteger和int值在多線程下遞增操作的測試
這篇文章主要介紹了Java中對AtomicInteger和int值在多線程下遞增操作的測試,本文得出AtomicInteger操作 與 int操作的效率大致相差在50-80倍上下的結(jié)論,需要的朋友可以參考下2014-09-09

