Reputation: 15351
I have a couple of questions with regards to Swing and using EDT for GUI updates. I just started reading on this stuff so I am a full beginner in this area:
SwingUtilities.invokeLater
we enqueue it to the current queue of GUI update tasks (the EDT) right? How exactly does the SwingWorker ensure that done()
method is run on EDT? It sets the following code:
future = new FutureTask<T>(callable) {
@Override
protected void done() {
doneEDT();
setState(StateValue.DONE);
}
};
so I was wondering whether FutureTask somehow makes sure that invokeLater
is called?
Thanks for all your answers.
Upvotes: 14
Views: 9551
Reputation: 36621
A good rule is that all operations (access/updates/...) should happen on the EDT. There are a few exceptions mentioned in the javadoc (certain methods of certain classes), but they are so hard to remember that it is easier to stick to the 'do everything on the EDT' approach. Exceptions will not be raised (luckily, JavaFX fixed this shortcoming). You can use a custom RepaintManager
to detect most of these violations: see this article.
Everything triggered by the user is handled on the EDT. For example if the user clicks on a button, the actionPerformed
of the corresponding Action
or ActionListener
will be called on the EDT.
Correct
The thing you schedule first will be executed first. The invokeLater
call simply adds the Runnable
at the end of the queue. Using invokeLater
a second time a bit later will add this new Runnable
after the previously scheduled Runnable
.
Take a look at the code for doneEDT
private void doneEDT() {
Runnable doDone =
new Runnable() {
public void run() {
done();
}
};
if (SwingUtilities.isEventDispatchThread()) {
doDone.run();
} else {
doSubmit.add(doDone);
}
}
Upvotes: 18
Reputation: 1
SwingWorker ensure that done() method is run on EDT via below code:
Runnable doDone =
new Runnable() {
public void run() {
done();
}
};
if (SwingUtilities.isEventDispatchThread()) {
doDone.run();
} else {
doSubmit.add(doDone);
}
Actually it add the doDone variable into AccumulativeRunnable doSubmit,
See the source code of AccumulativeRunnable.java you will find there has a below code
protected void submit() {
SwingUtilities.invokeLater(this);
}
That's why swingworker ensure the method done() running on EDT
Upvotes: 0
Reputation: 33544
1. In Java GUI applications, main()
method is not long lived, after scheduling the construction of GUI in the Event Dispatcher Thread
, the main()
method quits...and Now its EDT's responsibility to handle the GUI.
2. So we don't need to start our app on the EDT, its automatically done.
3 Always keep the UI work on the UI thread, and Non-UI work on the Non-UI thread.
So Always keep your EDT thread, which is the GUI thread only for GUI work.
Eg:
public static void main(String[] args){
EventQueue.invokeLater(new Runnable(){
public void run(){
myframe.setVisible(true);
}
}
}
4. Create a separate Non-UI thread to handle that long time taking method.
5. You can simply use a Thread
or use SwingWorker
which is specially introduced into Java to sync the UI and Non-UI thread.
6. SwingWorker doesnt ensure that done() method is run on EDT, but sync the output of it to EDT thread, which is the GUI thread.
Upvotes: -2
Reputation: 692231
doneEDT()
method has the following comment: Invokes done on the EDT.Upvotes: 8