java 中Executor, ExecutorService 和 Executors 間的不同
java 中Executor, ExecutorService 和 Executors 間的不同
java.util.concurrent.Executor, java.util.concurrent.ExecutorService, java.util.concurrent. Executors 這三者均是 Java Executor 框架的一部分,用來(lái)提供線程池的功能。因?yàn)閯?chuàng)建和管理線程非常心累,并且操作系統(tǒng)通常對(duì)線程數(shù)有限制,所以建議使用線程池來(lái)并發(fā)執(zhí)行任務(wù),而不是每次請(qǐng)求進(jìn)來(lái)時(shí)創(chuàng)建一個(gè)線程。使用線程池不僅可以提高應(yīng)用的響應(yīng)時(shí)間,還可以避免”java.lang.OutOfMemoryError: unable to create new native thread” 之類的錯(cuò)誤。
在 Java 1.5 時(shí),開(kāi)發(fā)者需要關(guān)心線程池的創(chuàng)建和管理,但在 Java 1.5 之后 Executor 框架提供了多種內(nèi)置的線程池,例如:FixedThreadPool(包含固定數(shù)目的線程),CachedThreadPool(可根據(jù)需要?jiǎng)?chuàng)建新的線程)等等。
Executor
Executor, ExecutorService, 和 Executors 最主要的區(qū)別是 Executor 是一個(gè)抽象層面的核心接口(大致代碼如下)。
public interface Executor {
void execute(Runnable command);
}
不同于 java.lang.Thread 類將任務(wù)和執(zhí)行耦合在一起, Executor 將任務(wù)本身和執(zhí)行任務(wù)分離,可以閱讀 difference between Thread and Executor 來(lái)了解 Thread 和 Executor 間更多的不同。
ExecutorService
ExecutorService 接口 對(duì) Executor 接口進(jìn)行了擴(kuò)展,提供了返回 Future 對(duì)象,終止,關(guān)閉線程池等方法。當(dāng)調(diào)用 shutDown 方法時(shí),線程池會(huì)停止接受新的任務(wù),但會(huì)完成正在 pending 中的任務(wù)。
Future 對(duì)象提供了異步執(zhí)行,這意味著無(wú)需等待任務(wù)執(zhí)行的完成,只要提交需要執(zhí)行的任務(wù),然后在需要時(shí)檢查 Future 是否已經(jīng)有了結(jié)果,如果任務(wù)已經(jīng)執(zhí)行完成,就可以通過(guò) Future.get() 方法獲得執(zhí)行結(jié)果。需要注意的是,F(xiàn)uture.get() 方法是一個(gè)阻塞式的方法,如果調(diào)用時(shí)任務(wù)還沒(méi)有完成,會(huì)等待直到任務(wù)執(zhí)行結(jié)束。
通過(guò) ExecutorService.submit() 方法返回的 Future 對(duì)象,還可以取消任務(wù)的執(zhí)行。Future 提供了 cancel() 方法用來(lái)取消執(zhí)行 pending 中的任務(wù)。
ExecutorService 部分代碼如下:
public interface ExecutorService extends Executor {
void shutdown();
<T> Future<T> submit(Callable<T> task);
<T> Future<T> submit(Runnable task, T result);
<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks, long timeout, TimeUnit unit) throws InterruptedException;
}
Executors
Executors 是一個(gè)工具類,類似于 Collections。提供工廠方法來(lái)創(chuàng)建不同類型的線程池,比如 FixedThreadPool 或 CachedThreadPool。
Executors 部分代碼:
public class Executors {
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads, 0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());
}
public static ExecutorService newCachedThreadPool() {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS, new SynchronousQueue<Runnable>());
}
}
下面詳細(xì)看一下三者的區(qū)別:
Executor vs ExecutorService vs Executors
正如上面所說(shuō),這三者均是 Executor 框架中的一部分。Java 開(kāi)發(fā)者很有必要學(xué)習(xí)和理解他們,以便更高效的使用 Java 提供的不同類型的線程池??偨Y(jié)一下這三者間的區(qū)別,以便大家更好的理解:
· Executor 和 ExecutorService 這兩個(gè)接口主要的區(qū)別是:ExecutorService 接口繼承了 Executor 接口,是 Executor 的子接口
· Executor 和 ExecutorService 第二個(gè)區(qū)別是:Executor 接口定義了 execute()方法用來(lái)接收一個(gè)Runnable接口的對(duì)象,而 ExecutorService 接口中的 submit()方法可以接受Runnable和Callable接口的對(duì)象。
· Executor 和 ExecutorService 接口第三個(gè)區(qū)別是 Executor 中的 execute() 方法不返回任何結(jié)果,而 ExecutorService 中的 submit()方法可以通過(guò)一個(gè) Future 對(duì)象返回運(yùn)算結(jié)果。
· Executor 和 ExecutorService 接口第四個(gè)區(qū)別是除了允許客戶端提交一個(gè)任務(wù),ExecutorService 還提供用來(lái)控制線程池的方法。比如:調(diào)用 shutDown() 方法終止線程池??梢酝ㄟ^(guò) 《Java Concurrency in Practice》 一書(shū)了解更多關(guān)于關(guān)閉線程池和如何處理 pending 的任務(wù)的知識(shí)。
· Executors 類提供工廠方法用來(lái)創(chuàng)建不同類型的線程池。比如: newSingleThreadExecutor() 創(chuàng)建一個(gè)只有一個(gè)線程的線程池,newFixedThreadPool(int numOfThreads)來(lái)創(chuàng)建固定線程數(shù)的線程池,newCachedThreadPool()可以根據(jù)需要?jiǎng)?chuàng)建新的線程,但如果已有線程是空閑的會(huì)重用已有線程。
總結(jié)
下表列出了 Executor 和 ExecutorService 的區(qū)別:

相關(guān)文章
Spring Boot實(shí)現(xiàn)Undertow服務(wù)器同時(shí)支持HTTP2、HTTPS的方法
這篇文章考慮如何讓Spring Boot應(yīng)用程序同時(shí)支持HTTP和HTTPS兩種協(xié)議。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12
mybatis動(dòng)態(tài)新增(insert)和修改(update)方式
這篇文章主要介紹了mybatis動(dòng)態(tài)新增(insert)和修改(update)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05
如何使用Java將word解析出來(lái)(包含格式和圖片)
今天遇到一個(gè)讀取word模板內(nèi)容的需求,下面這篇文章主要給大家介紹了關(guān)于如何使用Java將word解析出來(lái),包含格式和圖片,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-12-12
SpringBoot實(shí)現(xiàn)圖片防盜鏈功能
出于安全考慮,我們需要后端返回的圖片只允許在某個(gè)網(wǎng)站內(nèi)展示,不想被爬蟲(chóng)拿到圖片地址后被下載,或者,不想瀏覽器直接訪問(wèn)圖片鏈接,所以本文將給大家介紹SpringBoot實(shí)現(xiàn)圖片防盜鏈功能,需要的朋友可以參考下2024-04-04
Eclipse+Maven構(gòu)建Hadoop項(xiàng)目的方法步驟
這篇文章主要介紹了Eclipse+Maven構(gòu)建Hadoop項(xiàng)目的方法步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
java多線程批量拆分List導(dǎo)入數(shù)據(jù)庫(kù)的實(shí)現(xiàn)過(guò)程
這篇文章主要給大家介紹了關(guān)于java多線程批量拆分List導(dǎo)入數(shù)據(jù)庫(kù)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2021-10-10

