如何在java中取消ExecutorService

时间:2011-11-05 21:09:49

标签: java multithreading executorservice cancellation

我编写了一个使用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();
    }    
}

3 个答案:

答案 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 )