Java中为什么使用EventQueue.invokeLater
2017-04-04 18:08
465 查看
在Java中Swing是线程不安全的,是单线程的设计,这样的造成结果就是:只能从事件派发线程访问将要在屏幕上绘制的Swing组件。事件派发线程是调用paint和update等回调方法的线程,它还是事件监听器接口中定义的事件处理方法,例如,ActionListener中的actionPerformed方法在事件派发线程中调用。
Swing是事件驱动的,所以在回调函数中更新可见的GUI是很自然的事情,比如,有一个按钮被按下,项目列表需要更新时,则通常在与该按钮相关联的事件监听器的actionPerformed方法中来实现该列表的更新,从事件派发线程以外的线程中更新Swing组件是不正常的。
有时需要从事件派发线程以外的线程中更新Swing组件,例如,在actionPerformed中有很费时的操作,需要很长时间才能返回,按钮激活后需要很长时间才能看到更新的列表,按钮会长时间保持按下的状态只到actionPerformed返回,一般说来耗时的操作不应该在事件处理方法中执行,因为事件处理返回之前,其他事件是不能触发的,界面类似于卡住的状况,所以在独立的线程上执行比较耗时的操作可能更好,这会立即更新用户界面和释放事件派发线程去派发其他的事件。
SwingUtilities类提供了两个方法:invokeLate和invoteAndWait,它们都使事件派发线程上的可运行对象排队。当可运行对象排在事件派发队列的队首时,就调用其run方法。其效果是允许事件派发线程调用另一个线程中的任意一个代码块。
只有从事件派发线程才能更新组件。
程序示例:更新组件的错误方法
错误分析:在actionPerformed中,监听器把按钮的允许状态设置为false,由于是在事件派发线程上调用actionPerformed,所以setEnabled是一个有效的操作,但是在GetInfoThread中设置进度条是一个危险的做法,因为事件派发线程以外的线程更新了进度条,所以运行是不正常的。
invokeLater使用
Swing是事件驱动的,所以在回调函数中更新可见的GUI是很自然的事情,比如,有一个按钮被按下,项目列表需要更新时,则通常在与该按钮相关联的事件监听器的actionPerformed方法中来实现该列表的更新,从事件派发线程以外的线程中更新Swing组件是不正常的。
有时需要从事件派发线程以外的线程中更新Swing组件,例如,在actionPerformed中有很费时的操作,需要很长时间才能返回,按钮激活后需要很长时间才能看到更新的列表,按钮会长时间保持按下的状态只到actionPerformed返回,一般说来耗时的操作不应该在事件处理方法中执行,因为事件处理返回之前,其他事件是不能触发的,界面类似于卡住的状况,所以在独立的线程上执行比较耗时的操作可能更好,这会立即更新用户界面和释放事件派发线程去派发其他的事件。
SwingUtilities类提供了两个方法:invokeLate和invoteAndWait,它们都使事件派发线程上的可运行对象排队。当可运行对象排在事件派发队列的队首时,就调用其run方法。其效果是允许事件派发线程调用另一个线程中的任意一个代码块。
只有从事件派发线程才能更新组件。
程序示例:更新组件的错误方法
startButton.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { GetInfoThread t = new GetInfoThread(Test.this); t.start(); startButton.setEnabled(false); } }); class GetInfoThread extends Thread { Test applet; public GetInfoThread(Test applet) { this.applet = applet; } public void run() { while (true) { try { Thread.sleep(500); applet.getProgressBar().setValue(Math.random() * 100); } catch (InterruptedException e) { e.printStackTrace(); } } } }
错误分析:在actionPerformed中,监听器把按钮的允许状态设置为false,由于是在事件派发线程上调用actionPerformed,所以setEnabled是一个有效的操作,但是在GetInfoThread中设置进度条是一个危险的做法,因为事件派发线程以外的线程更新了进度条,所以运行是不正常的。
invokeLater使用
class GetInfoThread extends Thread { Test applet; Runnable runx; int value; public GetInfoThread(final Test applet) { this.applet = applet; runx = new Runnable() { public void run() { JProgressBar jpb = applet.getProgressBar(); jpb.setValue(value); } }; } public void run() { while (true) { try { Thread.sleep(500); value = (int) (Math.random() * 100); System.out.println(value); SwingUtilities.invokeLater(runx); } catch (InterruptedException e) { e.printStackTrace(); } } } }
相关文章推荐
- java.awt.EventQueue.invokeLater(new Runnable(){public void run(){}})(二)
- Java中EventQueue.invokeLater干什么用的
- 使用eventqueue.invokelater()好处、原因
- 使用eventqueue.invokelater()好处、原因
- java.awt.EventQueue.invokeLater(new Runnable(){public void run(){}})(一)
- Java中EventQueue.invokeLater干什么用的
- java.awt.EventQueue.invokeLater(new Runnable(){public void run(){}})(三)
- 使用eventqueue.invokelater()好处、原因
- Java中EventQueue.invokeLater干什么用的
- 使用eventqueue.invokelater()好处、原因
- EventQueue.invokeLater详解
- EventQueue.invokeLater(new Runnable())
- 关于为什么要加EventQueue.invokeAndWait的思考
- EventQueue.invokeLater(new Runnable())
- EventQueue.invokeLater
- EventQueue.invokeLater(new Runnable())
- EventQueue.invokeLater(new Runnable)
- 【转+分析】JAVA: 为什么要使用"抽象类"? 使用"抽象类"有什么好处?
- 为什么Java中要使用Checked Exceptions
- java中为什么要使用迭代器