Maison >Java >javaDidacticiel >Comment coordonner la réalisation de plusieurs SwingWorkers simultanés ?

Comment coordonner la réalisation de plusieurs SwingWorkers simultanés ?

Susan Sarandon
Susan Sarandonoriginal
2025-01-01 11:28:10642parcourir

How to Coordinate the Completion of Multiple Concurrent SwingWorkers?

En attente de plusieurs SwingWorkers

SwingWorker est un moyen pratique d'effectuer des tâches en arrière-plan et de mettre à jour l'interface graphique à partir du thread de travail. Cependant, il peut y avoir des situations où plusieurs instances SwingWorker s'exécutent simultanément et il devient nécessaire de coordonner leur achèvement.

Considérez le code suivant, dans lequel plusieurs SwingWorkers sont créés et exécutés en boucle :

import java.awt.FlowLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.lang.reflect.InvocationTargetException;
import javax.swing.*;

public class TestApplet extends JApplet {

    @Override
    public void init() {
        try {
            SwingUtilities.invokeAndWait(new Runnable() {
                @Override
                public void run() {
                    createGUI();
                }
            });
        } catch (InterruptedException | InvocationTargetException ex) {
        }
    }

    private void createGUI() {
        getContentPane().setLayout(new FlowLayout());
        JButton startButton = new JButton("Do work");
        startButton.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent ae) {
                for (int i = 0; i < 10; i++) {
                    new SwingWorker<Void, Void>() {
                        @Override
                        protected Void doInBackground() throws Exception {
                            // Do some work
                            return null;
                        }
                    }.execute();
                }
            }
        });
        getContentPane().add(startButton);
    }
}

Lorsque vous cliquez sur le bouton "Faire du travail", 10 instances SwingWorker sont créées et exécutées. Chaque SwingWorker effectue un travail en arrière-plan et met à jour l'interface graphique lorsque le travail est terminé. Cependant, il n'y a aucune coordination entre les SwingWorkers, ce qui signifie qu'ils peuvent effectuer n'importe quel ordre et peuvent entraîner une mise à jour erratique de l'interface graphique, surtout si les tâches sous-jacentes prennent plus ou moins de temps.

Pour garantir que les mises à jour de l'interface graphique de manière cohérente et prévisible, il est important de coordonner la réalisation des SwingWorkers. Plusieurs approches peuvent être utilisées pour y parvenir, et l'approche la plus adaptée dépend des exigences spécifiques de l'application.

1. Barrière

Une barrière est un mécanisme de synchronisation qui permet à plusieurs threads d'attendre jusqu'à ce qu'ils aient tous atteint un point commun. Dans le contexte de SwingWorkers, une barrière peut être utilisée pour garantir que l'interface graphique se met à jour uniquement une fois que tous les SwingWorkers ont terminé leurs tâches en procédant ainsi :

import java.lang.reflect.InvocationTargetException;
import java.util.concurrent.*;
import javax.swing.*;

public class TestApplet extends JApplet {

    private static final int NUM_WORKERS = 10;
    private static final ExecutorService EXECUTOR = Executors.newFixedThreadPool(NUM_WORKERS);
    private static final CyclicBarrier BARRIER = new CyclicBarrier(NUM_WORKERS, new Runnable() {

        @Override
        public void run() {
            // All SwingWorkers have completed, update the GUI
        }
    });

    @Override
    public void init() {
        try {
            SwingUtilities.invokeAndWait(new Runnable() {
                @Override
                public void run() {
                    createGUI();
                }
            });
        } catch (InterruptedException | InvocationTargetException ex) {
        }
    }

    private void createGUI() {
        JButton startButton = new JButton("Do work");
        startButton.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent ae) {
                for (int i = 0; i < NUM_WORKERS; i++) {
                    EXECUTOR.submit(new SwingWorker<Void, Void>() {

                        @Override
                        protected Void doInBackground() throws Exception {
                            // Do some work
                            return null;
                        }

                        @Override
                        protected void done() {
                            BARRIER.await();
                        }
                    });
                }
            }
        });
        getContentPane().add(startButton);
    }
}

Avec cette approche, les SwingWorkers exécutent leurs tâches simultanément, et la machine virtuelle Java (JVM) gère le pool de threads et planifie l'exécution des tâches. Lorsque tous les SwingWorkers ont terminé leurs tâches, la barrière est déclenchée et la méthode run() de la barrière est exécutée, qui est responsable de la mise à jour de l'interface graphique.

2. CountDownLatch

Un CountDownLatch est un autre mécanisme de synchronisation qui permet à plusieurs threads d'attendre qu'un certain nombre d'événements se produisent. Dans le contexte de SwingWorkers, un CountDownLatch peut être utilisé pour garantir que l'interface graphique se met à jour uniquement une fois que tous les SwingWorkers ont terminé leurs tâches, comme ceci :

import java.lang.reflect.InvocationTargetException;
import java.util.concurrent.*;
import javax.swing.*;

public class TestApplet extends JApplet {

    private static final int NUM_WORKERS = 10;
    private static final ExecutorService EXECUTOR = Executors.newFixedThreadPool(NUM_WORKERS);
    private static final CountDownLatch LATCH = new CountDownLatch(NUM_WORKERS);

    @Override
    public void init() {
        try {
            SwingUtilities.invokeAndWait(new Runnable() {
                @Override
                public void run() {
                    createGUI();
                }
            });
        } catch (InterruptedException | InvocationTargetException ex) {
        }
    }

    private void createGUI() {
        JButton startButton = new JButton("Do work");
        startButton.addActionListener(new ActionListener() {
            @Override
            public void actionPerformed(ActionEvent ae) {
                for (int i = 0; i < NUM_WORKERS; i++) {
                    EXECUTOR.submit(new SwingWorker<Void, Void>() {

                        @Override
                        protected Void doInBackground() throws Exception {
                            // Do some work
                            return null;
                        }

                        @Override
                        protected void done() {
                            LATCH.countDown();
                        }
                    });
                }
            }
        });
        getContentPane().add(startButton);
    }

    private void updateGUI() {
        // Update the GUI
    }

    @Override
    public void run() {
        try {
            LATCH.await();
            updateGUI();
        } catch (InterruptedException ex) {
        }
    }
}

Avec cette approche, les SwingWorkers exécutent leurs tâches. simultanément, et la JVM gère le pool de threads et planifie les tâches à exécuter. Lorsque tous les SwingWorkers ont terminé leurs tâches, le nombre de verrous atteint zéro, ce qui permet à la méthode run() de l'applet de continuer l'exécution et de mettre à jour l'interface graphique.

3. Phaser

Un Phaser est une synchronisation

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