无法停止使用 ExecutorService 启动的任务

2024-02-06

抱歉,我必须打开一个新线程来描述这个问题。

今天早上我问这个问题 https://stackoverflow.com/questions/8632773/how-to-stop-immediately-a-task-which-is-started-using-an-executorservice,有一些回复,但我的问题仍然没有解决。

这次我将附上一些可运行的代码(简化但具有相同的问题)供您重现问题:

public class ThreadPoolTest {
    public static void main(String[] args) throws Exception {
        final ExecutorService taskExecutor = Executors.newFixedThreadPool(5);
        Future<Void> futures[] = new Future[5];
        for (int i = 0; i < futures.length; ++i)
            futures[i] = startTask(taskExecutor);

        for (int i = 0; i < futures.length; ++i)
            System.out.println("futures[i].cancel(true): " + futures[i].cancel(true));

        System.out.println("Cancel DONE.");
        taskExecutor.shutdown();
    }

    private static Future<Void> startTask(final ExecutorService taskExecutor) {
        Future<Void> f = taskExecutor.submit(new Callable<Void>() {
            public Void call() throws Exception {
                try {
                    downloadFile(new URI("http://stackoverflow.com"));
                    while(true) {
                        System.out.println(Thread.currentThread().getName() + ": " + Thread.currentThread().isInterrupted());
                        if(Thread.currentThread().isInterrupted())
                            break;
                    }
                } catch (Exception ex) {
                    ex.printStackTrace();
                }
                return null;
            } 
        });
        return f;
    }

    private static void downloadFile (final URI uri) throws Exception {
//        if(true) return;
        Socket socket = new Socket (uri.getHost(), uri.getPort() == -1 ? 80 : uri.getPort());
        return;
    }
}

上面的代码很可能会陷入无限循环(您可能需要多次运行代码来见证我所看到的),正如您在我调用的 main 方法中看到的那样futures[i].cancel(true)对于所有任务,我不知道为什么会发生这种情况,这已经折磨了我一天多了。

对你的帮助表示感谢。


我玩过你的代码,注意到线程的中断状态有时在套接字创建之前为 true,而在创建之后为 false。

我尝试中断一个线程并调用 Socket 构造函数,但之后线程始终保持中断状态。我也尝试取消线程池的关闭,问题仍然发生。

然后我尝试使用 5 个不同的 URI,而不是始终使用相同的 URI。而且这个问题从未发生过。

所以我写了这个简单的程序,表明线程池不是罪魁祸首,而是套接字:

public static void main(String[] args) throws Exception {
    final URI uri = new URI("http://stackoverflow.com");
    for (int i = 0; i < 5; i++) {
        Runnable r = new Runnable() {
            @Override
            public void run() {
                Thread.currentThread().interrupt();
                System.out.println(Thread.currentThread().isInterrupted());
                try {
                    Socket socket = new Socket (uri.getHost(), uri.getPort() == -1 ? 80 : uri.getPort());
                }
                catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().isInterrupted());
            }
        };
        new Thread(r).start();
    }
}

事实上,当 5 个线程创建到同一主机和端口的套接字时,其中 4 个线程的中断状态被清除。

然后我尝试同步套接字创建(在单个锁上,但我猜您可能为每个主机/端口使用一个锁):

synchronized(lock) {
    try {
        Socket socket = new Socket (uri.getHost(), uri.getPort() == -1 ? 80 : uri.getPort());
    }
    catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

和 TADA...问题消失了。我会在 Oracle 中打开一个错误来表明该问题。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

无法停止使用 ExecutorService 启动的任务 的相关文章

随机推荐