Stare pytanie, ale myślę, że możesz przedłużyć ThreadPoolExecutor, aby przechwycić działające odwołanie do wątku przedExecute(). Po wywołaniu metody shutdownNow() możesz zatrzymać() wszystkie uruchomione wątki. Chociaż zdecydowanie zalecałbym poleganie na isInterrupted() w Twoich zadaniach.
Przykładowy kod ->
public class KillableThreadPoolExecutor extends ThreadPoolExecutor {
private final Map<Runnable, Thread> executingThreads;
public KillableThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, String threadNamePrefix) {
super(corePoolSize, maximumPoolSize, keepAliveTime, unit, new YoungMemorySafeLinkedBlockingQueue<Runnable>(), ThreadFactories.create(threadNamePrefix));
executingThreads = new HashMap<>(maximumPoolSize);
}
@Override
protected synchronized void beforeExecute(Thread t, Runnable r) {
super.beforeExecute(t, r);
executingThreads.put(r, t);
}
@Override
protected synchronized void afterExecute(Runnable r, Throwable t) {
super.afterExecute(r, t);
if(executingThreads.containsKey(r)) {
executingThreads.remove(r);
}
}
@Override
public synchronized List<Runnable> shutdownNow() {
List<Runnable> runnables = super.shutdownNow();
for(Thread t : executingThreads.values()) {
t.stop();
}
return runnables;
}
}
Dlaczego nie (lub nie możesz) użyć 'shtudown()' lub 'shutdownNow()'? To jest sposób na zrobienie tego. – Gray
Ponieważ musi on czekać na zakończenie bieżących zadań przed zakończeniem wątków. Nie zabija natychmiast nici. – deadlock
'shutdown()' czeka na zakończenie bieżących zadań. Właśnie o to chodzi. – Gray