JavaSwing多线程死锁问题解析

【独家特稿】在基于Java Swing进行图形界面开发的时候,经常遇到的就是Swing多线程问题。我们可以想想一下,如果需要在一个图形界面上显示很多数据,这些数据是经过长时间、复杂的查询和运算得到的。如果在图形界面的同一个线程中进行查询和运算工作则会导致一段时间界面处于死机状态,这会给用户带来不良的互动感受。为了解决这个问题,一般会单独启动一个线程进行运算和查询工作,并随时更新图形界面。这时候,另一个问题就出现了,可能不仅没有解决原来偶尔死机问题,还可能导致程序彻底死掉。幸运的是在JDK中暗藏了一个中断程序的快捷键,就是CTRL+BREAK,这个快捷键Sun并没有在文档中公布。如果在命令行模式下启动Java程序,然后按CTRL+BREAK键,会得到堆栈的跟踪信息。从这些跟踪信息中就可以知道具体引发死机的位置了。

为无锡等地区用户提供了全套网页设计制作服务,及无锡网站建设行业解决方案。主营业务为成都网站设计、网站建设、无锡网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!

当一个程序产生死锁的时候,你一定会希望尽快找到原因并且解决它。这时候,你一般的精力会用在查找引发死锁的位置,另一半的精力会用于对堆栈进行跟踪一确定引发死锁的原因。但是在Java Swing程序中,你的所有努力可能都是没有价值的。这是因为Java对Swing的多线程编程有一个特殊要求。就是在Swing里,只能在与Swing相同的线程里对GUI元件进行修改。

也就是说,如果你要执行类似于jLabel1.setText("blabla")代码,必须在Swing线程中,而不允许在其他线程当中。如果必须在其他线程中修改元件,可以使用类似一下方式解决:

 
 
 
  1. SwingUtilities.invokeLater(new Runnable() {  
  2.     public void run() {  
  3.         jLabel1.setText("blabla");  
  4.     }  

invokeLater方法虽然表面有时间延迟执行含义,但是实际上几乎没有任何影响,可能在几毫秒之内就会被执行。另外还有一个invokeAndWait方法,除非特殊需要,否则几乎是不用的。

在不使用invokeLater的情况下,导致刷新问题是可以理解的,但是导致死锁就优点令人匪夷所思了。幸运的是,不是任何时候都需要调用改方法,这是因为大多数情况下,我们都是在与Swing同一个线程里进行界面更新。例如监听按钮点击事件的ActionListener.actionPerformed方法就是运行在与Swing相同的线程中的。但是如果在回调类中引用了另一个类,并且是不属于AWT/Swing的,那么结果就很难确定了。所以说使用invokeLater应该是最安全的。

需要注意的是,在invokeLater做的任何事情,都会导致Swing线程窗口绘制工作暂停下来,等候invokeLater工作结束。所以不要在invokeLater进行耗时操作,尽量只执行那些界面绘制相关的工作。可以通过代码重构,将那些与界面更新相关的代码集中起来统一处理。

一个建议是那些在Swing中使用的类进行合理的设计。代码执行前判断是否处于Swing线程当中(使用SwingUtilities.isEventDispatchThread()方法),如果不是,则需要通过SwingUtilities.invokeLater(Runnable)执行,否则则直接执行代码。但是这说起来简单,但是实际操作会遇到很多困难。

【编辑推荐】

  1. 有趣的Java对象序列化缓存问题
  2. 关于Java对象序列化您不知道的5件事
  3. Java 7 I/O新功能探秘:同步操作,多播与随机存取
  4. Java实用技巧:当不能抛出checked异常时
  5. 多线程开发的捷径:构建Java并发模型框架

网站栏目:JavaSwing多线程死锁问题解析
文章源于:http://www.hantingmc.com/qtweb/news49/10249.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联