gpt4 book ai didi

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

转载 作者:塔克拉玛干 更新时间:2023-11-01 22:53:32 24 4
gpt4 key购买 nike

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

今天早上我问了this question ,有一些回复,但我的问题仍然没有解决。

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

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;
}
}

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

非常感谢您的帮助。

最佳答案

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

我试过中断线程并调用 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 中打开一个 bug 来表明问题。

关于java - 无法停止使用 ExecutorService 启动的任务,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8636908/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com