Home > Java > javaTutorial > How to Coordinate the Completion of Multiple Concurrent SwingWorkers?

How to Coordinate the Completion of Multiple Concurrent SwingWorkers?

Susan Sarandon
Release: 2025-01-01 11:28:10
Original
639 people have browsed it

How to Coordinate the Completion of Multiple Concurrent SwingWorkers?

Waiting for multiple SwingWorkers

SwingWorker is a convenient way to perform background tasks and update the GUI from the worker thread. However, there may be situations where multiple SwingWorker instances are running concurrently, and it becomes necessary to coordinate their completion.

Consider the following code, where multiple SwingWorkers are created and executed in a loop:

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);
    }
}
Copy after login

When the "Do work" button is clicked, 10 SwingWorker instances are created and executed. Each SwingWorker performs some work in the background and updates the GUI when the work is done. However, there is no coordination between the SwingWorkers, which means they can complete in any order and may cause the GUI to update erratically, especially if the underlying tasks take varying amounts of time.

To ensure that the GUI updates in a consistent and predictable manner, it is important to coordinate the completion of the SwingWorkers. Several approaches can be used to achieve this, and the most suitable approach depends on the specific requirements of the application.

1. Barrier

A barrier is a synchronization mechanism that allows multiple threads to wait until all of them have reached a common point. In the context of SwingWorkers, a barrier can be used to ensure that the GUI updates only after all SwingWorkers have completed their tasks by doing this:

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);
    }
}
Copy after login

With this approach, the SwingWorkers execute their tasks concurrently, and the Java Virtual Machine (JVM) manages the thread pool and schedules the tasks for execution. When all SwingWorkers have completed their tasks, the barrier is triggered, and the run() method of the barrier is executed, which is responsible for updating the GUI.

2. CountDownLatch

A CountDownLatch is another synchronization mechanism that allows multiple threads to wait until a certain number of events have occurred. In the context of SwingWorkers, a CountDownLatch can be used to ensure that the GUI updates only after all SwingWorkers have completed their tasks as it is done like this:

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) {
        }
    }
}
Copy after login

With this approach, the SwingWorkers execute their tasks concurrently, and the JVM manages the thread pool and schedules the tasks for execution. When all SwingWorkers have completed their tasks, the latch count reaches zero, which allows the run() method of the applet to continue execution and update the GUI.

3. Phaser

A Phaser is a synchronization

The above is the detailed content of How to Coordinate the Completion of Multiple Concurrent SwingWorkers?. For more information, please follow other related articles on the PHP Chinese website!

source:php.cn
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Latest Articles by Author
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template