Java swing 界面卡死问题的分析与解决 | LSABLOG

首页 » Program » Java » 正文

Java swing 界面卡死问题的分析与解决

0x00 问题概述

在用swing做一个项目的时候,发现程序在运行一个长时间任务的时候界面会卡死,直到任务完成。具体描述就是按下一个按钮执行任务,接着按钮就一直是按下的状态,界面卡死,直到这个任务完成才恢复正常。

 

0x01 问题分析

经过查阅资料,发现原来swing是线程不安全的,是单线程设计。Swing有初始线程,事件派发线程,工作线程。UI展示完毕之后初始线程就结束了,控制就交给了事件派发线程,这之后所有组件的事件行为都由事件派发线程处理,事件派发线程是调用paint和update等回调方法的线程,它还是事件监听器接口中定义的事件处理方法,也就是说只能从事件派发线程访问组件。这就是真相了,程序由事件派发线程执行一个任务,这时就要等待这个线程处理完这个任务,所以界面出现卡死现象,等这个任务结束了,事件派发线程就可以更新组件,就恢复正常了。所以,一些长时间的任务就要交给工作线程处理。

 

0x02 问题解决

最直接简单的方法就是把耗时长的任务单独开一个线程处理,即工作线程。

 

0x03 相关要点

要注意的是组件的更新(如进度条)最好别放在工作线程里,这是危险的,所以要单独开一个更新组件的线程,再把对象排到事件派发线程的队列中,这里要介绍两个函数:

SwingUtilities.invokeLater():当可运行对象排在事件派发队列的队首时,就调用其run方法。其效果是允许事件派发线程调用另一个线程中的任意一个代码块。把可运行的对象放入队列后就返回。

SwingUtilities.invokeAndWait():和invokeLater一样,但是它一直等待知道已启动了可运行的run方法才返回。如果一个操作在另外一个操作执行之前必须从一个组件获得信息,则应使用SwingUtilities.invokeAndWait()方法。

 

0x04 参考资料

developer.51cto.com/art/200907/136290.htm

blog.csdn.net/sjf8013/article/details/7693669

blog.csdn.net/zhenshiyiqie/article/details/7605533

 

 

 

Comment