java swing里面大部分类都不是线程安全的,如果通过多个线程去操作swing对象,很可能会出现很多诡异的现象,如果你想让它变成线程安全的,就需要用一个特殊的线程去操作swing对象,也就是EDT线程,也就是事件调度线程(Event Dispatch Thread,EDT)
一个GUI 程序有很多线程同时运行,其中有一个叫做 事件调度线程(EDT),用来处理我们在程序里面所有的回调(最常见的就是我们点击按钮后执行的actionPerformed方法),所有的操作Swing对象的操作必须放在这个线程里面,否则就会出问题
看一个例子
import java.awt.BorderLayout; import javax.swing.JFrame; import javax.swing.JTextField; import javax.swing.SwingUtilities; /** *NotInEDTSample
just demonstrates the usage of Swing EDT simply. * * @author Jimmy.haung(SZ Team) * @since DUI (Mar 25, 2013) */ public class NotInEDTSample extends JFrame { private static final long serialVersionUID = 1L; private JTextField m_txt; public NotInEDTSample() { initGUI(); notInEDT(); } /** * Init the GUI */ public void initGUI() { this.setTitle("a simple EDT Sample"); m_txt = new JTextField(); getContentPane().add(m_txt, BorderLayout.CENTER); } /** * Process not under the EDT. 这里我启动了10个线程来改变m_txt
的内容. */ private void notInEDT() { for (int i = 0; i < 4; ++i) { new Thread(new Runnable() { @Override public void run() { while (true) { m_txt.setText("我不在EDT中操作!"); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }).start(); } } /*private void notInEDT() { for (int i = 0; i < 4; ++i) { new Thread(new Runnable() { @Override public void run() { while (true) { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { m_txt.setText("我在EDT中操作!"); } }); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }).start(); } }*/ /** * Launch the application. */ public static void main(String[] args) { SwingUtilities.invokeLater(new Runnable() { public void run() { try { NotInEDTSample oFrame = new NotInEDTSample(); oFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); oFrame.setLocationRelativeTo(null); oFrame.setSize(300, 200); oFrame.setVisible(true); } catch (Exception e) { e.printStackTrace(); } } }); } }
我们在notInEDT用其他线程去改变Text的值,应该很快就会出问题,而且没有抛出异常
只要我们把下面的这个样子,让EDT线程去处理就没有问题了
private void notInEDT() { for (int i = 0; i < 4; ++i) { new Thread(new Runnable() { @Override public void run() { while (true) { SwingUtilities.invokeLater(new Runnable() { @Override public void run() { m_txt.setText("我在EDT中操作!"); } }); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } } }).start(); } }
这里调用了invokeLater方法把UI操作丢给EDT线程去处理,看看说明
让一个Runnable接口的run方法能够异步在EDT线程里面执行,等于把这个操作假如到一个队列队尾,等前面的操作都完成了再执行这个操作
/** * Causes doRun.run() to be executed asynchronously on the * AWT event dispatching thread. This will happen after all * pending AWT events have been processed. This method should * be used when an application thread needs to update the GUI. * In the following example theinvokeLater
call queues * theRunnable
objectdoHelloWorld
* on the event dispatching thread and * then prints a message. ** Runnable doHelloWorld = new Runnable() { * public void run() { * System.out.println("Hello World on " + Thread.currentThread()); * } * }; * * SwingUtilities.invokeLater(doHelloWorld); * System.out.println("This might well be displayed before the other message."); ** If invokeLater is called from the event dispatching thread -- * for example, from a JButton's ActionListener -- the doRun.run() will * still be deferred until all pending events have been processed. * Note that if the doRun.run() throws an uncaught exception * the event dispatching thread will unwind (not the current thread). ** Additional documentation and examples for this method can be * found in * How to Use Threads, * in The Java Tutorial. *
* As of 1.3 this method is just a cover for
java.awt.EventQueue.invokeLater()
. ** Unlike the rest of Swing, this method can be invoked from any thread. * * @see #invokeAndWait */ public static void invokeLater(Runnable doRun)
有一个可以功能类似但是有区别的方法,invokeAndWait方法,功能和invokeLater差不多,也是可以把run方法放到EDT里面执行,但是区别在于是同步的,并且不能在EDT里面被调用
/** * CausesdoRun.run()
to be executed synchronously on the * AWT event dispatching thread. This call blocks until * all pending AWT events have been processed and (then) *doRun.run()
returns. This method should * be used when an application thread needs to update the GUI. * It shouldn't be called from the event dispatching thread. * Here's an example that creates a new application thread * that usesinvokeAndWait
to print a string from the event * dispatching thread and then, when that's finished, print * a string from the application thread. ** final Runnable doHelloWorld = new Runnable() { * public void run() { * System.out.println("Hello World on " + Thread.currentThread()); * } * }; * * Thread appThread = new Thread() { * public void run() { * try { * SwingUtilities.invokeAndWait(doHelloWorld); * } * catch (Exception e) { * e.printStackTrace(); * } * System.out.println("Finished on " + Thread.currentThread()); * } * }; * appThread.start(); ** Note that if theRunnable.run
method throws an * uncaught exception * (on the event dispatching thread) it's caught and rethrown, as * anInvocationTargetException
, on the caller's thread. ** Additional documentation and examples for this method can be * found in * How to Use Threads, * in The Java Tutorial. *
* As of 1.3 this method is just a cover for *
java.awt.EventQueue.invokeAndWait()
. * * @exception InterruptedException if we're interrupted while waiting for * the event dispatching thread to finish excecuting *doRun.run()
* @exception InvocationTargetException if an exception is thrown * while runningdoRun
* * @see #invokeLater */ public static void invokeAndWait(final Runnable doRun) throws InterruptedException, InvocationTargetException
下面我来系统分析一下为什么不能在EDT里面调用 invokeAndWait
首先需要理解java swing的event 队列,我们对UI的基本所有操作都会生成一个event,加入到event队列里面,由EDT线程来逐一处理,比如我们鼠标点击一个按钮,就是把注册在按钮里面的事件加入event里面去处理
我们来比较invokeLater和invokeAndWait的源码区别
/** * invokeLater将我们的runnable包装成事件丢进eventQueue就没有管了 */ public static void invokeLater(Runnable runnable) { Toolkit.getEventQueue().postEvent( new InvocationEvent(Toolkit.getDefaultToolkit(), runnable)); }
static void invokeAndWait(Object source, Runnable runnable) throws InterruptedException, InvocationTargetException { if (EventQueue.isDispatchThread()) {//这里是防止在EDT里面调用 throw new Error("Cannot call invokeAndWait from the event dispatcher thread"); } class AWTInvocationLock {} Object lock = new AWTInvocationLock(); InvocationEvent event = new InvocationEvent(source, runnable, lock, true); //这里是关键,当线程进入这里之后获得了锁,然后wait了 synchronized (lock) { Toolkit.getEventQueue().postEvent(event); lock.wait(); } Throwable eventThrowable = event.getThrowable(); if (eventThrowable != null) { throw new InvocationTargetException(eventThrowable); } }
invokeAndWait的时候wait了,然后看在是什么地方notify的,InvocationEvent里面
/** * Executes the Runnable'srun()
method and notifies the * notifier (if any) whenrun()
has returned or thrown an exception. * * @see #isDispatched */ public void dispatch() { try { if (catchExceptions) { try { runnable.run(); } catch (Throwable t) { if (t instanceof Exception) { exception = (Exception) t; } throwable = t; } } else { runnable.run(); } } finally { dispatched = true; if (notifier != null) { synchronized (notifier) { notifier.notifyAll();//执行完我们的操作后,notify所有线程 } } } }
这就是为什么说invokeAndWait这个方法是同步的原因,调用这个方法的线程会一直堵塞知道执行完毕
现在假如我们在EDT里面调用invokeAndWait,务必会造成死锁,看看下一面这个例子
import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JTextField; import javax.swing.SwingUtilities; public class TestAction extends JFrame { private static final long serialVersionUID = -7462155330900531124L; private JButton jb1 = new JButton("确定"); private JTextField txt = new JTextField(10); public TestAction() { jb1.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { String name = ((JButton) e.getSource()).getText(); txt.setText(name); } }); setLayout(null); add(txt); add(jb1); txt.setBounds(50, 100, 200, 30); jb1.setBounds(270, 100, 70, 30); } public static void main(String[] args) { SwingUtilities.invokeLater(new Runnable() { public void run() { SwingConsole.run(new TestAction(), 500, 500); } }); } }
import javax.swing.*; public class SwingConsole { public static void run(final JFrame f, final int width, final int height) { SwingUtilities.invokeLater(new Runnable() { public void run() { f.setTitle(f.getClass().getSimpleName()); f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); f.setSize(width, height); f.setVisible(true); } }); } }
这样写是没有错误的~~假如我们用invokeAndWait,肯定就死锁了
import java.lang.reflect.InvocationTargetException; import javax.swing.*; public class SwingConsole { public static void run(final JFrame f, final int width, final int height){ try { SwingUtilities.invokeAndWait(new Runnable(){ public void run(){ f.setTitle(f.getClass().getSimpleName()); f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); f.setSize(width, height); f.setVisible(true); } }); } catch (InvocationTargetException e) { e.printStackTrace(); } catch (InterruptedException e) { e.printStackTrace(); } } }
这部分还有一个重点,不要在EDT里面调用费时间的操作,这样会造成界面卡主
来看一个例子,我们模拟10个文件有序上传,并且显示进度条
import java.awt.BorderLayout; import java.awt.FlowLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.*; public class TestProgress extends Thread implements ActionListener { private static JProgressBar progressBar; JFrame jf = new JFrame("Test"); JPanel jp = new JPanel(); JTextArea jta = new JTextArea(); JButton jb = new JButton("点击"); public static void main(String[] args) { new TestProgress(); } public TestProgress() { jp.setLayout(new FlowLayout()); progressBar = new JProgressBar(); progressBar.setValue(0); progressBar.setStringPainted(true); jf.add(jp, BorderLayout.NORTH); jf.add(new JScrollPane(jta)); jp.add(progressBar); jp.add(jb); jf.add(new JScrollPane(jta), BorderLayout.CENTER); jb.addActionListener(this); jf.setSize(300, 200); jf.setLocation(300, 200); jf.setVisible(true); jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); } @Override public void run() { for (int i = 0; i < 10;i++) {//10个文件 UpLordTread lordTread = new UpLordTread(progressBar,jta,"文件" + i); lordTread.start();//启动上传程序 try { lordTread.join();//这就是关键~~等待这个线程完成 } catch (InterruptedException e) { e.printStackTrace(); } } } public void actionPerformed(ActionEvent e) { String comm = e.getActionCommand(); if ("点击".equals(comm)) { this.start();//不能在EDT线程里面执行费时的操作,防止UI卡死 jb.setEnabled(false); } } } /** * 文件上传线程 * @author yellowbaby * */ class UpLordTread extends Thread{ JTextArea jta; JProgressBar progressBar; public UpLordTread(JProgressBar progressBar,JTextArea jta,String fileName) { super(fileName); this.jta = jta; this.progressBar = progressBar; } public void run() { for (int i = 0; i <= 100; i++) { progressBar.setValue(i); String temp = Thread.currentThread().getName() + ":" + i + "\n"; jta.append(temp); try { Thread.sleep(10); } catch (Exception ee) { ee.printStackTrace(); } } progressBar.setValue(0); } }
我们点击按钮后,我并没有,在actionPerformaed里面直接调用上传的循环代码,而是从新开了一个线程,去执行,这是为什么呢?
因为actionPerformaed的代码是由EDT调用的,如果这个方法不立即返回的话,EDT线程就无法去处理其他事件,界面也就卡死了
可以试试把代码改成这样,点击按钮后界面就会卡死,然后就只有等到全部上传后界面才恢复
import java.awt.BorderLayout; import java.awt.FlowLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.*; public class TestProgress extends Thread implements ActionListener { private static JProgressBar progressBar; JFrame jf = new JFrame("Test"); JPanel jp = new JPanel(); JTextArea jta = new JTextArea(); JButton jb = new JButton("点击"); public static void main(String[] args) { new TestProgress(); } public TestProgress() { jp.setLayout(new FlowLayout()); progressBar = new JProgressBar(); progressBar.setValue(0); progressBar.setStringPainted(true); jf.add(jp, BorderLayout.NORTH); jf.add(new JScrollPane(jta)); jp.add(progressBar); jp.add(jb); jf.add(new JScrollPane(jta), BorderLayout.CENTER); jb.addActionListener(this); jf.setSize(300, 200); jf.setLocation(300, 200); jf.setVisible(true); jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); } @Override public void run() { for (int i = 0; i < 10;i++) {//10个文件 UpLordTread lordTread = new UpLordTread(progressBar,jta,"文件" + i); lordTread.start();//启动上传程序 try { lordTread.join();//这就是关键~~等待这个线程完成 } catch (InterruptedException e) { e.printStackTrace(); } } } public void actionPerformed(ActionEvent e) { String comm = e.getActionCommand(); if ("点击".equals(comm)) { //this.start();//不能在EDT线程里面执行费时的操作,防止UI卡死 for (int i = 0; i < 10;i++) {//10个文件 UpLordTread lordTread = new UpLordTread(progressBar,jta,"文件" + i); lordTread.start();//启动上传程序 try { lordTread.join();//这就是关键~~等待这个线程完成 } catch (InterruptedException e1) { e1.printStackTrace(); } } jb.setEnabled(false); } } } /** * 文件上传线程 * @author yellowbaby * */ class UpLordTread extends Thread{ JTextArea jta; JProgressBar progressBar; public UpLordTread(JProgressBar progressBar,JTextArea jta,String fileName) { super(fileName); this.jta = jta; this.progressBar = progressBar; } public void run() { for (int i = 0; i <= 100; i++) { progressBar.setValue(i); String temp = Thread.currentThread().getName() + ":" + i + "\n"; jta.append(temp); try { Thread.sleep(10); } catch (Exception ee) { ee.printStackTrace(); } } progressBar.setValue(0); } }
所以我们的费时操作都要丢到背后线程去处理,JDK里面有一个SwingWork可以帮我们处理这个问题
import java.awt.BorderLayout; import java.awt.FlowLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.*; public class TestProgress extends Thread implements ActionListener { private static JProgressBar progressBar; JFrame jf = new JFrame("Test"); JPanel jp = new JPanel(); JTextArea jta = new JTextArea(); JButton jb = new JButton("点击"); public static void main(String[] args) { new TestProgress(); } public TestProgress() { jp.setLayout(new FlowLayout()); progressBar = new JProgressBar(); progressBar.setValue(0); progressBar.setStringPainted(true); jf.add(jp, BorderLayout.NORTH); jf.add(new JScrollPane(jta)); jp.add(progressBar); jp.add(jb); jf.add(new JScrollPane(jta), BorderLayout.CENTER); jb.addActionListener(this); jf.setSize(300, 200); jf.setLocation(300, 200); jf.setVisible(true); jf.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); } public void actionPerformed(ActionEvent e) { String comm = e.getActionCommand(); if ("点击".equals(comm)) { SwingWorkerswingWorker = new SwingWorker () { @Override protected Void doInBackground() throws Exception { for (int i = 0; i < 10; i++) {// 10个文件 UpLordTread lordTread = new UpLordTread(progressBar, jta, "文件" + i); lordTread.start();// 启动上传程序 try { lordTread.join();// 这就是关键~~等待这个线程完成 } catch (InterruptedException e) { e.printStackTrace(); } } return null; } @Override protected void done() { System.out.println("上传成功"); } }; swingWorker.execute(); jb.setEnabled(false); } } } /** * 文件上传线程 * * @author yellowbaby * */ class UpLordTread extends Thread { JTextArea jta; JProgressBar progressBar; public UpLordTread(JProgressBar progressBar, JTextArea jta, String fileName) { super(fileName); this.jta = jta; this.progressBar = progressBar; } public void run() { for (int i = 0; i <= 100; i++) { progressBar.setValue(i); String temp = Thread.currentThread().getName() + ":" + i + "\n"; jta.append(temp); try { Thread.sleep(10); } catch (Exception ee) { ee.printStackTrace(); } } progressBar.setValue(0); } }
SwingWork里面有两个主要的方法,doInBackground和done,doInBackground就是我们的背后线程,done是做完doInBackground后调用的,主要用来更新UI