com.jidesoft.utils.SwingWorker Maven / Gradle / Ivy
/*
* $Id: SwingWorker.java,v 1.5 2007/03/01 19:55:54 idk Exp $
*
* Copyright (c) 1995, 2006, Oracle and/or its affiliates. All rights reserved.
* ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.
*/
package com.jidesoft.utils;
import javax.swing.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.util.List;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;
/**
* An abstract class to perform lengthy GUI-interacting tasks in a dedicated thread.
*
*
* When writing a multi-threaded application using Swing, there are two constraints to keep in mind: (refer to How to Use Threads for more details):
* - Time-consuming tasks should not be run on the Event Dispatch Thread. Otherwise the application
* becomes unresponsive.
- Swing components should be accessed on the Event Dispatch Thread only.
*
*
*
*
*
* These constraints mean that a GUI application with time intensive computing needs at least two threads: 1) a thread
* to perform the lengthy task and 2) the Event Dispatch Thread (EDT) for all GUI-related activities. This
* involves inter-thread communication which can be tricky to implement.
*
*
* {@code SwingWorker} is designed for situations where you need to have a long running task run in a background thread
* and provide updates to the UI either when done, or while processing. Subclasses of {@code SwingWorker} must implement
* the {@see #doInBackground} method to perform the background computation.
*
*
*
* Workflow
*
* There are three threads involved in the life cycle of a {@code SwingWorker} : -
*
* Current thread: The {@link #execute} method is called on this thread. It schedules {@code SwingWorker} for the
* execution on a worker thread and returns immediately. One can wait for the {@code SwingWorker} to complete
* using the {@link #get get} methods.
-
*
* Worker thread: The {@link #doInBackground} method is called on this thread. This is where all background
* activities should happen. To notify {@code PropertyChangeListeners} about bound properties changes use the {@link
* #firePropertyChange firePropertyChange} and {@link #getPropertyChangeSupport} methods. By default there are two bound
* properties available: {@code state} and {@code progress}.
-
*
* Event Dispatch Thread: All Swing related activities occur on this thread. {@code SwingWorker} invokes the
* {@link #process process} and {@link #done} methods and notifies any {@code PropertyChangeListeners} on this thread.
*
*
*
* Often, the Current thread is the Event Dispatch Thread.
*
*
*
* Before the {@code doInBackground} method is invoked on a worker thread, {@code SwingWorker} notifies any
* {@code PropertyChangeListeners} about the {@code state} property change to {@code StateValue.STARTED}. After the
* {@code doInBackground} method is finished the {@code done} method is executed. Then {@code SwingWorker} notifies any
* {@code PropertyChangeListeners} about the {@code state} property change to {@code StateValue.DONE}.
*
*
* {@code SwingWorker} is only designed to be executed once. Executing a {@code SwingWorker} more than once will not
* result in invoking the {@code doInBackground} method twice.
*
*
* Sample Usage
*
* The following example illustrates the simplest use case. Some processing is done in the background and when done you
* update a Swing component.
*
*
* Say we want to find the "Meaning of Life" and display the result in a {@code JLabel}.
*
*
* final JLabel label;
* class MeaningOfLifeFinder extends SwingWorker<String, Object> {
* {@code @Override}
* public String doInBackground() {
* return findTheMeaningOfLife();
* }
*
* {@code @Override}
* protected void done() {
* try {
* label.setText(get());
* } catch (Exception ignore) {
* }
* }
* }
*
* (new MeaningOfLifeFinder()).execute();
*
*
*
* The next example is useful in situations where you wish to process data as it is ready on the Event Dispatch
* Thread.
*
*
* Now we want to find the first N prime numbers and display the results in a {@code JTextArea}. While this is
* computing, we want to update our progress in a {@code JProgressBar}. Finally, we also want to print the prime
* numbers to {@code System.out}.
*
* class PrimeNumbersTask extends
* SwingWorker<List<Integer>, Integer> {
* PrimeNumbersTask(JTextArea textArea, int numbersToFind) {
* //initialize
* }
*
* {@code @Override}
* public List<Integer> doInBackground() {
* while (! enough && ! isCancelled()) {
* number = nextPrimeNumber();
* publish(number);
* setProgress(100 * numbers.size() / numbersToFind);
* }
* }
* return numbers;
* }
*
* {@code @Override}
* protected void process(List<Integer> chunks) {
* for (int number : chunks) {
* textArea.append(number + "\n");
* }
* }
* }
*
* JTextArea textArea = new JTextArea();
* final JProgressBar progressBar = new JProgressBar(0, 100);
* PrimeNumbersTask task = new PrimeNumbersTask(textArea, N);
* task.addPropertyChangeListener(
* new PropertyChangeListener() {
* public void propertyChange(PropertyChangeEvent evt) {
* if ("progress".equals(evt.getPropertyName())) {
* progressBar.setValue((Integer)evt.getNewValue());
* }
* }
* });
*
* task.execute();
* System.out.println(task.get()); //prints all prime numbers we have got
*
*
*
* Because {@code SwingWorker} implements {@code Runnable}, a {@code SwingWorker} can be submitted to an {@link
* java.util.concurrent.Executor} for execution.
*
* @param the result type returned by this {@code SwingWorker's} {@code doInBackground} and {@code get} methods
* @param the type used for carrying out intermediate results by this {@code SwingWorker's} {@code publish} and
* {@code process} methods
* @author Igor Kushnirskiy
* @version $Revision: 1.5 $ $Date: 2007/03/01 19:55:54 $
*/
public abstract class SwingWorker implements Future, Runnable {
/**
* number of worker threads.
*/
private static final int MAX_WORKER_THREADS = 10;
/**
* current progress.
*/
private volatile int progress;
/**
* current state.
*/
private volatile StateValue state;
/**
* everything is run inside this FutureTask. Also it is used as a delegate for the Future API.
*/
private final FutureTask future;
/**
* all propertyChangeSupport goes through this.
*/
private final PropertyChangeSupport propertyChangeSupport;
/**
* handler for {@code process} method.
*/
private AccumulativeRunnable doProcess;
/**
* handler for progress property change notifications.
*/
private AccumulativeRunnable doNotifyProgressChange;
private static final AccumulativeRunnable doSubmit =
new DoSubmitAccumulativeRunnable();
private static ExecutorService executorService = null;
/**
* Values for the {@code state} bound property.
*/
public enum StateValue {
/**
* Initial {@code SwingWorker} state.
*/
PENDING,
/**
* {@code SwingWorker} is {@code STARTED} before invoking {@code doInBackground}.
*/
STARTED,
/**
* {@code SwingWorker} is {@code DONE} after {@code doInBackground} method is finished.
*/
DONE
}
;
/**
* Constructs this {@code SwingWorker}.
*/
public SwingWorker() {
Callable callable =
new Callable() {
public T call() throws Exception {
setState(StateValue.STARTED);
return doInBackground();
}
};
future = new FutureTask(callable) {
@Override
protected void done() {
doneEDT();
setState(StateValue.DONE);
}
};
state = StateValue.PENDING;
propertyChangeSupport = new SwingWorkerPropertyChangeSupport(this);
doProcess = null;
doNotifyProgressChange = null;
}
/**
* Computes a result, or throws an exception if unable to do so.
*
*
* Note that this method is executed only once.
*
*
* Note: this method is executed in a background thread.
*
* @return the computed result
*
* @throws Exception if unable to compute a result
*/
protected abstract T doInBackground() throws Exception;
/**
* Sets this {@code Future} to the result of computation unless it has been cancelled.
*/
public final void run() {
future.run();
}
/**
* Sends data chunks to the {@link #process} method. This method is to be used from inside the {@code
* doInBackground} method to deliver intermediate results for processing on the Event Dispatch Thread inside
* the {@code process} method.
*
*
* Because the {@code process} method is invoked asynchronously on the Event Dispatch Thread multiple
* invocations to the {@code publish} method might occur before the {@code process} method is executed. For
* performance purposes all these invocations are coalesced into one invocation with concatenated arguments.
*
*
* For example:
*
*
* publish("1");
* publish("2", "3");
* publish("4", "5", "6");
*
*
* might result in:
*
*
* process("1", "2", "3", "4", "5", "6")
*
*
*
* Sample Usage. This code snippet loads some tabular data and updates {@code DefaultTableModel} with it.
* Note that it safe to mutate the tableModel from inside the {@code process} method because it is invoked on the
* Event Dispatch Thread.
*
*
* class TableSwingWorker extends
* SwingWorker<DefaultTableModel, Object[]> {
* private final DefaultTableModel tableModel;
*
* public TableSwingWorker(DefaultTableModel tableModel) {
* this.tableModel = tableModel;
* }
*
* {@code @Override}
* protected DefaultTableModel doInBackground() throws Exception {
* for (Object[] row = loadData();
* ! isCancelled() && row != null;
* row = loadData()) {
* publish((Object[]) row);
* }
* return tableModel;
* }
*
* {@code @Override}
* protected void process(List<Object[]> chunks) {
* for (Object[] row : chunks) {
* tableModel.addRow(row);
* }
* }
* }
*
*
* @param chunks intermediate results to process
* @see #process
*/
protected final void publish(V... chunks) {
synchronized (this) {
if (doProcess == null) {
doProcess = new AccumulativeRunnable() {
@Override
public void run(List args) {
process(args);
}
@Override
protected void submit() {
doSubmit.add(this);
}
};
}
}
doProcess.add(chunks);
}
/**
* Receives data chunks from the {@code publish} method asynchronously on the Event Dispatch Thread.
*
*
* Please refer to the {@link #publish} method for more details.
*
* @param chunks intermediate results to process
* @see #publish
*/
protected void process(List chunks) {
}
/**
* Executed on the Event Dispatch Thread after the {@code doInBackground} method is finished. The default
* implementation does nothing. Subclasses may override this method to perform completion actions on the Event
* Dispatch Thread. Note that you can query status inside the implementation of this method to determine the
* result of this task or whether this task has been cancelled.
*
* @see #doInBackground
* @see #isCancelled()
* @see #get
*/
protected void done() {
}
/**
* Sets the {@code progress} bound property. The value should be from 0 to 100.
*
*
* Because {@code PropertyChangeListener}s are notified asynchronously on the Event Dispatch Thread multiple
* invocations to the {@code setProgress} method might occur before any {@code PropertyChangeListeners} are invoked.
* For performance purposes all these invocations are coalesced into one invocation with the last invocation
* argument only.
*
*
* For example, the following invocations:
*
*
* setProgress(1);
* setProgress(2);
* setProgress(3);
*
*
* might result in a single {@code PropertyChangeListener} notification with the value {@code 3}.
*
* @param progress the progress value to set
* @throws IllegalArgumentException is value not from 0 to 100
*/
protected final void setProgress(int progress) {
if (progress < 0 || progress > 100) {
throw new IllegalArgumentException("the value should be from 0 to 100");
}
if (this.progress == progress) {
return;
}
int oldProgress = this.progress;
this.progress = progress;
if (!getPropertyChangeSupport().hasListeners("progress")) {
return;
}
synchronized (this) {
if (doNotifyProgressChange == null) {
doNotifyProgressChange =
new AccumulativeRunnable() {
@Override
public void run(List args) {
firePropertyChange("progress",
args.get(0),
args.get(args.size() - 1));
}
@Override
protected void submit() {
doSubmit.add(this);
}
};
}
}
doNotifyProgressChange.add(oldProgress, progress);
}
/**
* Returns the {@code progress} bound property.
*
* @return the progress bound property.
*/
public final int getProgress() {
return progress;
}
/**
* Schedules this {@code SwingWorker} for execution on a worker thread. There are a number of worker
* threads available. In the event all worker threads are busy handling other {@code SwingWorkers} this
* {@code SwingWorker} is placed in a waiting queue.
*
*
* Note: {@code SwingWorker} is only designed to be executed once. Executing a {@code SwingWorker} more than once
* will not result in invoking the {@code doInBackground} method twice.
*/
public final void execute() {
getWorkersExecutorService().execute(this);
}
// Future methods START
/**
* {@inheritDoc}
*/
public final boolean cancel(boolean mayInterruptIfRunning) {
return future.cancel(mayInterruptIfRunning);
}
/**
* {@inheritDoc}
*/
public final boolean isCancelled() {
return future.isCancelled();
}
/**
* {@inheritDoc}
*/
public final boolean isDone() {
return future.isDone();
}
/**
* {@inheritDoc}
*
* Note: calling {@code get} on the Event Dispatch Thread blocks all events, including repaints, from
* being processed until this {@code SwingWorker} is complete.
*
*
* When you want the {@code SwingWorker} to block on the Event Dispatch Thread we recommend that you use a
* modal dialog.
*
*
* For example:
*
*
* class SwingWorkerCompletionWaiter extends PropertyChangeListener {
* private JDialog dialog;
*
* public SwingWorkerCompletionWaiter(JDialog dialog) {
* this.dialog = dialog;
* }
*
* public void propertyChange(PropertyChangeEvent event) {
* if ("state".equals(event.getPropertyName())
* && SwingWorker.StateValue.DONE == event.getNewValue()) {
* dialog.setVisible(false);
* dialog.dispose();
* }
* }
* }
* JDialog dialog = new JDialog(owner, true);
* swingWorker.addPropertyChangeListener(
* new SwingWorkerCompletionWaiter(dialog));
* swingWorker.execute();
* //the dialog will be visible until the SwingWorker is done
* dialog.setVisible(true);
*
*/
public final T get() throws InterruptedException, ExecutionException {
return future.get();
}
/**
* {@inheritDoc}
*
* Please refer to {@link #get} for more details.
*/
public final T get(long timeout, TimeUnit unit) throws InterruptedException,
ExecutionException, TimeoutException {
return future.get(timeout, unit);
}
// Future methods END
// PropertyChangeSupports methods START
/**
* Adds a {@code PropertyChangeListener} to the listener list. The listener is registered for all properties. The
* same listener object may be added more than once, and will be called as many times as it is added. If {@code
* listener} is {@code null}, no exception is thrown and no action is taken.
*
*
* Note: This is merely a convenience wrapper. All work is delegated to {@code PropertyChangeSupport} from {@link
* #getPropertyChangeSupport}.
*
* @param listener the {@code PropertyChangeListener} to be added
*/
public final void addPropertyChangeListener(PropertyChangeListener listener) {
getPropertyChangeSupport().addPropertyChangeListener(listener);
}
/**
* Removes a {@code PropertyChangeListener} from the listener list. This removes a {@code PropertyChangeListener}
* that was registered for all properties. If {@code listener} was added more than once to the same event source, it
* will be notified one less time after being removed. If {@code listener} is {@code null}, or was never added, no
* exception is thrown and no action is taken.
*
*
* Note: This is merely a convenience wrapper. All work is delegated to {@code PropertyChangeSupport} from {@link
* #getPropertyChangeSupport}.
*
* @param listener the {@code PropertyChangeListener} to be removed
*/
public final void removePropertyChangeListener(PropertyChangeListener listener) {
getPropertyChangeSupport().removePropertyChangeListener(listener);
}
/**
* Reports a bound property update to any registered listeners. No event is fired if {@code old} and {@code new} are
* equal and non-null.
*
*
* This {@code SwingWorker} will be the source for any generated events.
*
*
* When called off the Event Dispatch Thread {@code PropertyChangeListeners} are notified asynchronously on
* the Event Dispatch Thread.
*
* Note: This is merely a convenience wrapper. All work is delegated to {@code PropertyChangeSupport} from {@link
* #getPropertyChangeSupport}.
*
* @param propertyName the programmatic name of the property that was changed
* @param oldValue the old value of the property
* @param newValue the new value of the property
*/
public final void firePropertyChange(String propertyName, Object oldValue,
Object newValue) {
getPropertyChangeSupport().firePropertyChange(propertyName,
oldValue, newValue);
}
/**
* Returns the {@code PropertyChangeSupport} for this {@code SwingWorker}. This method is used when flexible access
* to bound properties support is needed.
*
* This {@code SwingWorker} will be the source for any generated events.
*
*
* Note: The returned {@code PropertyChangeSupport} notifies any {@code PropertyChangeListener}s asynchronously on
* the Event Dispatch Thread in the event that {@code firePropertyChange} or {@code
* fireIndexedPropertyChange} are called off the Event Dispatch Thread.
*
* @return {@code PropertyChangeSupport} for this {@code SwingWorker}
*/
public final PropertyChangeSupport getPropertyChangeSupport() {
return propertyChangeSupport;
}
// PropertyChangeSupports methods END
/**
* Returns the {@code SwingWorker} state bound property.
*
* @return the current state
*/
public final StateValue getState() {
/*
* DONE is a special case
* to keep getState and isDone is sync
*/
if (isDone()) {
return StateValue.DONE;
}
else {
return state;
}
}
/**
* Sets this {@code SwingWorker} state bound property.
*
* @param state the state to set
*/
private void setState(StateValue state) {
StateValue old = this.state;
this.state = state;
firePropertyChange("state", old, state);
}
/**
* Invokes {@code done} on the EDT.
*/
private void doneEDT() {
Runnable doDone =
new Runnable() {
public void run() {
done();
}
};
if (SwingUtilities.isEventDispatchThread()) {
doDone.run();
}
else {
doSubmit.add(doDone);
}
}
/**
* returns workersExecutorService.
*
* returns the service stored in the appContext or creates it if necessary. If the last one it triggers autoShutdown
* thread to get started.
*
* @return ExecutorService for the {@code SwingWorkers}
*/
private static synchronized ExecutorService getWorkersExecutorService() {
if (executorService == null) {
//this creates non-daemon threads.
ThreadFactory threadFactory =
new ThreadFactory() {
final AtomicInteger threadNumber = new AtomicInteger(1);
public Thread newThread(final Runnable r) {
StringBuilder name =
new StringBuilder("SwingWorker-pool-");
name.append(System.identityHashCode(this));
name.append("-thread-");
name.append(threadNumber.getAndIncrement());
Thread t = new Thread(r, name.toString());
;
if (t.isDaemon())
t.setDaemon(false);
if (t.getPriority() != Thread.NORM_PRIORITY)
t.setPriority(Thread.NORM_PRIORITY);
return t;
}
};
/*
* We want a to have no more than MAX_WORKER_THREADS
* running threads.
*
* We want a worker thread to wait no longer than 1 second
* for new tasks before terminating.
*/
executorService = new ThreadPoolExecutor(0, MAX_WORKER_THREADS,
5L, TimeUnit.SECONDS,
new LinkedBlockingQueue(),
threadFactory) {
private final ReentrantLock pauseLock = new ReentrantLock();
private final Condition unpaused = pauseLock.newCondition();
private boolean isPaused = false;
private final ReentrantLock executeLock = new ReentrantLock();
@Override
public void execute(Runnable command) {
/*
* ThreadPoolExecutor first tries to run task
* in a corePool. If all threads are busy it
* tries to add task to the waiting queue. If it
* fails it run task in maximumPool.
*
* We want corePool to be 0 and
* maximumPool to be MAX_WORKER_THREADS
* We need to change the order of the execution.
* First try corePool then try maximumPool
* pool and only then store to the waiting
* queue. We can not do that because we would
* need access to the private methods.
*
* Instead we enlarge corePool to
* MAX_WORKER_THREADS before the execution and
* shrink it back to 0 after.
* It does pretty much what we need.
*
* While we changing the corePoolSize we need
* to stop running worker threads from accepting new
* tasks.
*/
//we need atomicity for the execute method.
executeLock.lock();
try {
pauseLock.lock();
try {
isPaused = true;
}
finally {
pauseLock.unlock();
}
setCorePoolSize(MAX_WORKER_THREADS);
super.execute(command);
setCorePoolSize(0);
pauseLock.lock();
try {
isPaused = false;
unpaused.signalAll();
}
finally {
pauseLock.unlock();
}
}
finally {
executeLock.unlock();
}
}
@Override
protected void afterExecute(Runnable r, Throwable t) {
super.afterExecute(r, t);
pauseLock.lock();
try {
while (isPaused) {
unpaused.await();
}
}
catch (InterruptedException ignore) {
}
finally {
pauseLock.unlock();
}
}
};
}
return executorService;
}
private static class DoSubmitAccumulativeRunnable
extends AccumulativeRunnable implements ActionListener {
private static final int DELAY = (int) (1000 / 30);
@Override
protected void run(List args) {
for (Runnable runnable : args) {
runnable.run();
}
}
@Override
protected void submit() {
Timer timer = new Timer(DELAY, this);
timer.setRepeats(false);
timer.start();
}
public void actionPerformed(ActionEvent event) {
run();
}
}
private class SwingWorkerPropertyChangeSupport
extends PropertyChangeSupport {
SwingWorkerPropertyChangeSupport(Object source) {
super(source);
}
@Override
public void firePropertyChange(final PropertyChangeEvent evt) {
if (SwingUtilities.isEventDispatchThread()) {
super.firePropertyChange(evt);
}
else {
doSubmit.add(
new Runnable() {
public void run() {
SwingWorkerPropertyChangeSupport.this
.firePropertyChange(evt);
}
});
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy