Maison >Java >javaDidacticiel >Comment puis-je implémenter un délai d'attente pour les tâches pouvant être interrompues dans un ExecutorService ?

Comment puis-je implémenter un délai d'attente pour les tâches pouvant être interrompues dans un ExecutorService ?

DDD
DDDoriginal
2024-12-11 14:54:11903parcourir

How Can I Implement a Timeout for Interruptable Tasks in an ExecutorService?

ExecutorService avec délai d'attente pour les tâches pouvant être interrompues

Lors de l'exécution simultanée de tâches, il est souvent souhaitable de disposer d'un mécanisme permettant d'interrompre progressivement les tâches qui dépassent un délai spécifié. temps mort. Ceci est particulièrement utile dans les situations où des tâches de longue durée peuvent entraîner des blocages d'applications ou des problèmes de performances.

Implémentation d'un TimeoutExecutorService

Voici une implémentation d'un TimeoutThreadPoolExecutor qui étend la ThreadPoolExecutor standard et intègre un délai d'attente fonctionnalité :

class TimeoutThreadPoolExecutor extends ThreadPoolExecutor {

    private final long timeout;
    private final TimeUnit timeoutUnit;

    private final ScheduledExecutorService timeoutExecutor = Executors.newSingleThreadScheduledExecutor();
    private final ConcurrentMap<Runnable, ScheduledFuture> runningTasks = new ConcurrentHashMap<>();

    public TimeoutThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, long timeout, TimeUnit timeoutUnit) {
        super(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue);
        this.timeout = timeout;
        this.timeoutUnit = timeoutUnit;
    }

    // Override methods to implement timeout behavior

    @Override
    protected void beforeExecute(Thread t, Runnable r) {
        if (timeout > 0) {
            ScheduledFuture scheduled = timeoutExecutor.schedule(new TimeoutTask(t), timeout, timeoutUnit);
            runningTasks.put(r, scheduled);
        }
    }

    @Override
    protected void afterExecute(Runnable r, Throwable t) {
        ScheduledFuture timeoutTask = runningTasks.remove(r);
        if (timeoutTask != null) {
            timeoutTask.cancel(false);
        }
    }

    // Timeout task to interrupt threads
    class TimeoutTask implements Runnable {

        private final Thread thread;

        public TimeoutTask(Thread thread) {
            this.thread = thread;
        }

        @Override
        public void run() {
            thread.interrupt();
        }
    }
}

Utilisation :

Pour utiliser le TimeoutThreadPoolExecutor, vous pouvez simplement l'instancier avec une valeur de délai d'attente :

TimeoutThreadPoolExecutor executor = new TimeoutThreadPoolExecutor(
    4, // corePoolSize
    8, // maximumPoolSize
    1, // keepAliveTime
    TimeUnit.SECONDS, // timeUnit
    new LinkedBlockingQueue<>(), // workQueue
    5, // timeout
    TimeUnit.SECONDS // timeoutUnit
);

Ensuite , soumettez les tâches à l'exécuteur testamentaire comme d'habitude :

executor.submit(() -> {
    // long-running task
});

Si une tâche prend plus de temps supérieur au délai d'expiration spécifié, le thread exécutant la tâche sera interrompu, provoquant la fin de la tâche correctement.

Solution alternative

Une autre approche pour implémenter un délai d'attente pour une tâche consiste à utiliser ScheduledExecutorService comme suggéré dans la réponse. Cela implique de soumettre la tâche en tant que Callable et de conserver le futur créé. Une deuxième tâche peut alors être programmée pour annuler le futur après une certaine période, interrompant ainsi la tâche.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn