Java使用ExecutorService來停止線程服務(wù)
使用ExecutorService來停止線程服務(wù)
之前的文章中我們提到了ExecutorService可以使用shutdown和shutdownNow來關(guān)閉。
這兩種關(guān)閉的區(qū)別在于各自的安全性和響應(yīng)性。shutdownNow強(qiáng)行關(guān)閉速度更快,但是風(fēng)險(xiǎn)也更大,因?yàn)槿蝿?wù)可能正在執(zhí)行的過程中被結(jié)束了。而shutdown正常關(guān)閉雖然速度比較慢,但是卻更安全,因?yàn)樗恢钡鹊疥?duì)列中的所有任務(wù)都執(zhí)行完畢之后才關(guān)閉。
使用shutdown
我們先看一個(gè)使用shutdown的例子:
public void useShutdown() throws InterruptedException {
ExecutorService executor = Executors.newFixedThreadPool(10);
Runnable runnableTask = () -> {
try {
TimeUnit.MILLISECONDS.sleep(300);
} catch (InterruptedException e) {
e.printStackTrace();
}
};
executor.submit(runnableTask);
executor.shutdown();
executor.awaitTermination(800, TimeUnit.MILLISECONDS);
}
awaitTermination將會阻塞直到所有正在執(zhí)行的任務(wù)完成,或者達(dá)到指定的timeout時(shí)間。
使用shutdownNow
當(dāng)通過shutdownNow來強(qiáng)行關(guān)閉ExecutorService是, 它會嘗試取消正在執(zhí)行的任務(wù),并返回所有已經(jīng)提交但是還沒有開始的任務(wù)。從而可以將這些任務(wù)保存起來,以便以后進(jìn)行處理。
但是這樣我們只知道了還沒有開始執(zhí)行的任務(wù),對于那些已經(jīng)開始執(zhí)行但是沒有執(zhí)行完畢卻被取消的任務(wù)我們無法獲取。
我們看下如何獲得開始執(zhí)行但是還沒有執(zhí)行完畢的任務(wù):
public class TrackingExecutor extends AbstractExecutorService {
private final ExecutorService executorService;
private final Set<Runnable> taskCancelledAtShutdown= Collections.synchronizedSet(new HashSet<Runnable>());
public TrackingExecutor(ExecutorService executorService){
this.executorService=executorService;
}
@Override
public void shutdown() {
executorService.shutdown();
}
@Override
public List<Runnable> shutdownNow() {
return executorService.shutdownNow();
}
@Override
public boolean isShutdown() {
return executorService.isShutdown();
}
@Override
public boolean isTerminated() {
return executorService.isTerminated();
}
@Override
public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException {
return executorService.awaitTermination(timeout,unit);
}
@Override
public void execute(Runnable command) {
executorService.execute(() -> {
try {
command.run();
}finally {
if(isShutdown() && Thread.currentThread().isInterrupted()){
taskCancelledAtShutdown.add(command);
}
}
});
}
public List<Runnable> getCancelledTask(){
if(! executorService.isTerminated()){
throw new IllegalStateException("executorService is not terminated");
}
return new ArrayList<>(taskCancelledAtShutdown);
}
}
上面的例子中我們構(gòu)建了一個(gè)新的ExecutorService,他傳入一個(gè)ExecutorService,并對其進(jìn)行封裝。
我們重寫了execute方法,在執(zhí)行完畢判斷該任務(wù)是否被中斷,如果被中斷則將其添加到CancelledTask列表中。
并提供一個(gè)getCancelledTask方法來返回未執(zhí)行完畢的任務(wù)。
我們看下怎么使用:
public void useShutdownNow() throws InterruptedException {
TrackingExecutor trackingExecutor=new TrackingExecutor(Executors.newCachedThreadPool());
Runnable runnableTask = () -> {
try {
TimeUnit.MILLISECONDS.sleep(300);
} catch (InterruptedException e) {
e.printStackTrace();
}
};
trackingExecutor.submit(runnableTask);
List<Runnable> notrunList=trackingExecutor.shutdownNow();
if(trackingExecutor.awaitTermination(800, TimeUnit.SECONDS)){
List<Runnable> runButCancelledList= trackingExecutor.getCancelledTask();
}
}
trackingExecutor.shutdownNow()返回的是未執(zhí)行的任務(wù)。而trackingExecutor.getCancelledTask()返回的是被取消的任務(wù)。
上面的任務(wù)其實(shí)還有一個(gè)缺點(diǎn),因?yàn)槲覀冊诖鎯Ρ蝗∠娜蝿?wù)列表的額時(shí)候taskCancelledAtShutdown.add(command),因?yàn)橹暗呐袛嗖皇窃硬僮鳎瑒t可能會產(chǎn)生誤報(bào)。
本文的例子請參考https://github.com/ddean2009/learn-java-concurrency/tree/master/ExecutorServiceShutdown
到此這篇關(guān)于Java使用ExecutorService來停止線程服務(wù)的文章就介紹到這了,更多相關(guān)Java ExecutorService停止線程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用volatile關(guān)鍵字的注意事項(xiàng)
volatile關(guān)鍵字是Java中的一種稍弱的同步機(jī)制,為什么稱之為弱機(jī)制。這篇文章主要介紹了Java使用volatile關(guān)鍵字的注意事項(xiàng),需要的朋友可以參考下2017-02-02
關(guān)于Spring?Boot內(nèi)存泄露排查的記錄
這篇文章主要介紹了關(guān)于Spring?Boot內(nèi)存泄露排查的記錄,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06
Springboot從配置文件properties讀取字符串亂碼的解決
這篇文章主要介紹了Springboot從配置文件properties讀取字符串亂碼的解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
Spring Boot修改內(nèi)置Tomcat默認(rèn)端口號的示例
本篇文章主要介紹了Spring Boot修改內(nèi)置Tomcat端口號的示例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08
Java案例之HashMap集合存儲學(xué)生對象并遍歷
這篇文章主要介紹了Java案例之HashMap集合存儲學(xué)生對象并遍歷,創(chuàng)建一個(gè)HashMap集合,鍵是學(xué)號(String),值是學(xué)生對象(Student),存儲三個(gè)鍵值對元素并遍歷,下文具體操作需要的朋友可以參考一下2022-04-04
利用數(shù)組實(shí)現(xiàn)棧(Java實(shí)現(xiàn))
這篇文章主要為大家詳細(xì)介紹了利用數(shù)組實(shí)現(xiàn)棧,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-09-09

