Heim > Java > javaLernprogramm > Wie koordiniere ich den Abschluss mehrerer gleichzeitiger SwingWorker?

Wie koordiniere ich den Abschluss mehrerer gleichzeitiger SwingWorker?

Susan Sarandon
Freigeben: 2025-01-01 11:28:10
Original
650 Leute haben es durchsucht

How to Coordinate the Completion of Multiple Concurrent SwingWorkers?

Warten auf mehrere SwingWorker

SwingWorker ist eine praktische Möglichkeit, Hintergrundaufgaben auszuführen und die GUI vom Worker-Thread aus zu aktualisieren. Es kann jedoch Situationen geben, in denen mehrere SwingWorker-Instanzen gleichzeitig ausgeführt werden und es notwendig wird, deren Abschluss zu koordinieren.

Betrachten Sie den folgenden Code, in dem mehrere SwingWorker erstellt und in einer Schleife ausgeführt werden:

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);
    }
}
Nach dem Login kopieren

Wenn auf die Schaltfläche „Arbeit ausführen“ geklickt wird, werden 10 SwingWorker-Instanzen erstellt und ausgeführt. Jeder SwingWorker führt einige Arbeiten im Hintergrund aus und aktualisiert die GUI, wenn die Arbeit erledigt ist. Es gibt jedoch keine Koordination zwischen den SwingWorkern, was bedeutet, dass sie in beliebiger Reihenfolge abschließen können und dazu führen können, dass die GUI unregelmäßig aktualisiert wird, insbesondere wenn die zugrunde liegenden Aufgaben unterschiedlich viel Zeit in Anspruch nehmen.

Um sicherzustellen, dass die GUI aktualisiert wird Es ist wichtig, die Fertigstellung der SwingWorkers konsistent und vorhersehbar zu koordinieren. Um dies zu erreichen, können mehrere Ansätze verwendet werden, und der am besten geeignete Ansatz hängt von den spezifischen Anforderungen der Anwendung ab.

1. Barriere

Eine Barriere ist ein Synchronisationsmechanismus, der es mehreren Threads ermöglicht, zu warten, bis alle einen gemeinsamen Punkt erreicht haben. Im Kontext von SwingWorkern kann eine Barriere verwendet werden, um sicherzustellen, dass die GUI erst aktualisiert wird, nachdem alle SwingWorker ihre Aufgaben abgeschlossen haben, indem Sie Folgendes tun:

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);
    }
}
Nach dem Login kopieren

Bei diesem Ansatz führen die SwingWorker ihre Aufgaben gleichzeitig aus und Die Java Virtual Machine (JVM) verwaltet den Thread-Pool und plant die Ausführung der Aufgaben. Wenn alle SwingWorker ihre Aufgaben erledigt haben, wird die Barriere ausgelöst und die run()-Methode der Barriere ausgeführt, die für die Aktualisierung der GUI verantwortlich ist.

2. CountDownLatch

Ein CountDownLatch ist ein weiterer Synchronisationsmechanismus, der es mehreren Threads ermöglicht, zu warten, bis eine bestimmte Anzahl von Ereignissen aufgetreten ist. Im Kontext von SwingWorkers kann ein CountDownLatch verwendet werden, um sicherzustellen, dass die GUI erst aktualisiert wird, nachdem alle SwingWorker ihre Aufgaben abgeschlossen haben, wie folgt:

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) {
        }
    }
}
Nach dem Login kopieren

Mit diesem Ansatz führen die SwingWorker ihre Aufgaben aus gleichzeitig, und die JVM verwaltet den Thread-Pool und plant die Ausführung der Aufgaben. Wenn alle SwingWorker ihre Aufgaben abgeschlossen haben, erreicht die Latch-Zählung Null, wodurch die run()-Methode des Applets die Ausführung fortsetzen und die GUI aktualisieren kann.

3. Phaser

Ein Phaser ist eine Synchronisation

Das obige ist der detaillierte Inhalt vonWie koordiniere ich den Abschluss mehrerer gleichzeitiger SwingWorker?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Neueste Artikel des Autors
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage