Ответ 1
Q: Когда именно начинается EDT? Какая строка кода отвечает [f] этого?
Внутренняя работа Swing является специфичной для JVM. Различные JVM запускают Event Dispatch Thread (EDT) на основе разных критериев. В общем, хотя:
EDT начинается, когда он получает свой первый
AWTEvent
.
Следы стека ниже подтверждают эту точку. Возьмем, например, следующий метод main
.
public static void main(String[] args) {
JFrame frame = new JFrame();
frame.setVisible(true);
}
В приведенном выше примере строка кода, ответственного за запуск EDT, составляет frame.setVisible(true);
Вышеупомянутый метод main
был выполнен на двух разных JVM. Точка останова была помещена в EventQueue.initDispatchThread
. Когда была удалена точка останова, были отмечены следующие следы стека.
Использование Mac JDK в потоке AWT-AppKit
:
EventQueue.initDispatchThread() line: 906
EventQueue.wakeup(boolean) line: 1109
NativeMethodAccessorImpl.invoke0(Method, Object, Object[]) line: not available [native method]
NativeMethodAccessorImpl.invoke(Object, Object[]) line: 39
DelegatingMethodAccessorImpl.invoke(Object, Object[]) line: 25
Method.invoke(Object, Object...) line: 597
SunToolkit.wakeupEventQueue(EventQueue, boolean) line: 348
PostEventQueue.postEvent(AWTEvent) line: 2137
SunToolkit.postEvent(AppContext, AWTEvent) line: 583
SunToolkit.executeOnEventHandlerThread(PeerEvent) line: 654
SunToolkit.executeOnEventHandlerThread(Object, Runnable) line: 631
EventFactoryProxy.windowMoved(CWindow) line: 89
Использование Oracle JDK для Windows в потоке main
:
java.awt.EventQueue.initDispatchThread() line: 861
java.awt.EventQueue.postEventPrivate(java.awt.AWTEvent) line: 199
java.awt.EventQueue.postEvent(java.awt.AWTEvent) line: 180
javax.swing.RepaintManager.scheduleProcessingRunnable(sun.awt.AppContext) line: 1369
javax.swing.RepaintManager.nativeAddDirtyRegion(sun.awt.AppContext, java.awt.Container, int, int, int, int) line: 548
javax.swing.SwingPaintEventDispatcher.createPaintEvent(java.awt.Component, int, int, int, int) line: 45
sun.awt.windows.WFramePeer(sun.awt.windows.WComponentPeer).postPaintIfNecessary(int, int, int, int) line: 741
sun.awt.windows.WFramePeer(sun.awt.windows.WComponentPeer).handlePaint(int, int, int, int) line: 736
sun.java2d.d3d.D3DScreenUpdateManager.repaintPeerTarget(sun.awt.windows.WComponentPeer) line: 274
sun.java2d.d3d.D3DScreenUpdateManager.createScreenSurface(sun.awt.Win32GraphicsConfig, sun.awt.windows.WComponentPeer, int, boolean) line: 175
...
sun.awt.windows.WToolkit.createFrame(java.awt.Frame) line: 383
javax.swing.JFrame(java.awt.Frame).addNotify() line: 460
javax.swing.JFrame(java.awt.Window).show() line: 859
javax.swing.JFrame(java.awt.Component).show(boolean) line: 1584
javax.swing.JFrame(java.awt.Component).setVisible(boolean) line: 1536
javax.swing.JFrame(java.awt.Window).setVisible(boolean) line: 842
Example.main(java.lang.String[]) line: 113
На Mac выполняется вызов PostEventQueue.postEvent(AWTEvent)
. Аналогично в Windows выполняется вызов java.awt.EventQueue.postEvent(java.awt.AWTEvent)
. Оба они в конечном итоге назовут EventQueue.initDispatchThread
.
В качестве другого примера рассмотрим следующий метод main
:
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
System.out.println("Start!");
}
});
}
Использование Mac JDK в потоке main
:
EventQueue.initDispatchThread() line: 906 [local variables unavailable]
EventQueue.postEventPrivate(AWTEvent) line: 227
EventQueue.postEvent(AWTEvent) line: 208
EventQueue.invokeLater(Runnable) line: 1048
SwingUtilities.invokeLater(Runnable) line: 1267
Example.main(String[]) line: 31
Использование Oracle JDK для Windows в потоке main
:
java.awt.EventQueue.initDispatchThread() line: 861
java.awt.EventQueue.postEventPrivate(java.awt.AWTEvent) line: 199
java.awt.EventQueue.postEvent(java.awt.AWTEvent) line: 180
java.awt.EventQueue.invokeLater(java.lang.Runnable) line: 999
javax.swing.SwingUtilities.invokeLater(java.lang.Runnable) line: 1267
Ответ на SwingUtilties.invokeLater
отвечает за запуск EDT. Здесь также вызываются вызовы EventQueue.postEvent(AWTEvent)
.
Мысли о "Мое предположение:" someSwingComponent.setVisible(true) "делает трюк, но я не уверен".
Не только любой вызов someSwingComponent.setVisible(true)
запустит EDT. Например, выполнение следующего метода main
не создает поток AWT-Event-Queue-0
:
public static void main(String[] args) {
JLabel label = new JLabel();
label.setVisible(true);
}
Ресурсы
Конечно, в EDT есть много ресурсов в Интернете.
- "Тема события Dispatch" из Учебников Java "Урок: Concurrency в Swing"
- "Threads and Swing" из Sun Developer Network
- "Swing Threads" из Java Glossary на mindprod.com
- "Как работает поток отправки событий?" в "Переполнение стека"