我编写了一个使用ExecutorService
运行某些线程的应用程序,并一直等到它们完成:
ExecutorService exService;
exService = Executors.newCachedThreadPool();
exService.execute(T1);
exService.execute(T2);
exService.shutdown();
boolean finshed = exService.awaitTermination(5, TimeUnit.MINUTES);
有时我需要取消这些线程的执行(整个ExecutorService
)
我尝试exService.shutdownNow()
,但它会抛出java.lang.InterruptedException
并且不会取消线程
如何取消这些线程的执行?
编辑:添加了作为nanda请求的T1类代码
public class TC implements Runnable{
private ExtractedDataBuffer Buffer;
private Scraper scraper;
private String AppPath;
private boolean Succeed=false;
private Map<String,Object> Result=null;
private JLabel StatElement;
public TC(ExtractedDataBuffer Buffer,String AppPath,String XMLfile,JLabel Stat) throws FileNotFoundException {
this.Buffer = Buffer;
this.AppPath=AppPath;
this.StatElement=Stat;
ScraperConfiguration config;
config = new ScraperConfiguration(AppPath + Main.XMLFilesPath +XMLfile);
scraper = new Scraper(config, AppPath);
}
private void extract(){
try{
mainF.SetIconStat("working", this.StatElement);
scraper.execute();
if(scraper.getStatus()==Scraper.STATUS_FINISHED){
this.Succeed=true;
Map<String,Object> tmp=new HashMap<String,Object>();
tmp.put("UpdateTime", ((Variable) scraper.getContext().get("UpdateTime")).toString().trim());
Buffer.setVal(this.Result);
mainF.SetIconStat("done", this.StatElement);
}else{
this.Succeed=false;
this.Result=null;
Buffer.setVal(null);
mainF.SetIconStat("error", this.StatElement);
}
}catch(Exception ex){
this.Succeed=false;
this.Result=null;
Buffer.setVal(null);
mainF.SetIconStat("error", this.StatElement);
}
}
public void run() {
this.extract();
}
}
答案 0 :(得分:5)
如果将shutdown()更改为shutdownNow(),则在编写的代码中执行的操作正确。但是,请查看shutdownNow()的文档:
除了努力尝试停止处理之外,没有任何保证 积极执行任务。例如,典型的实现方式 通过Thread.interrupt()取消,因此任何无法响应的任务 中断可能永远不会终止。
因此,您的T1和T2可能没有以正确的方式编码,并且不能很好地响应中断。你可以为他们复制代码吗?
-
根据你的代码,我猜需要很长时间的代码是scraper.execute(),对吧?所以在这些方法中,你必须经常检查这样的东西:
if (Thread.interrupted()) {
throw new InterruptedException();
}
如果中断到来,将在catch语句中抛出并捕获InterruptedException,线程将停止。
答案 1 :(得分:4)
我对Future#cancel()
的问题是后来对get()
的调用会抛出CancellationException
。有时我仍然希望能够在get()
上调用Future
,以便在关闭执行程序服务后检索部分结果。在这种情况下,可以在您提交给执行程序服务的可调用对象中实现终止逻辑。使用像
private volatile boolean terminate;
public void terminate() {
terminate = true;
}
并根据需要经常检查callable中的terminate
。此外,您必须在某处记住您的callables,以便可以在所有这些地方调用terminate()
。
答案 2 :(得分:1)
使用Executor。submit方法,它通过创建并返回可用于取消执行和/或等待的Future
来扩展基本方法Executor.execute(java.lang.Runnable)。完成:
task = Executor.submit( T1 )
...
task.cancel( true )