日常使用软件的过程中,偶尔会遇到软件突然卡住,再点击几次就变成“未响应”的情况。

在JavaFX应用中同样也会出现这种情况,在开发过程中应该尽量避免这种情况的出现。

>> 更多JavaFX文章 >> JavaFX桌面应用开发系列文章

1. “未响应”重现

应用程序出现“未响应”这种情况往往是因为在UI线程中处理一些耗时的业务,当UI线程在处理耗时的业务时,UI就会卡住。

下面通过一个示例(获取Google页面title信息)来演示一下“未响应”这种情况。

这里使用 jsoup 来抓取Google页面的title信息,需要引入jsoup的maven依赖:

<dependency>
<groupId>org.jsoup</groupId>
<artifactId>jsoup</artifactId>
<version>1.13.1</version>
</dependency>

编译AppService,实现抓取Google页面的title信息。

public class AppService {
public static final AppService INSTANCE = new AppService();
private AppService() {
}
public String visitGoogle() {
try {
Document document = Jsoup.parse(new URL("https://www.google.com"), 10_000);
return document.head().getElementsByTag("title").get(0).text();
} catch (Exception e) {
return e.getMessage();
}
}
}

因为没有F墙,这里访问Google肯定是超时的,这里设置了超时10秒。

接着改造AppUI,当点击Go按钮的时候,调用visitGoogle并将结果显示在界面上。

public class AppUI implements Initializable {
public Label text;
private AppService appService = AppService.INSTANCE;
private AppModel model = new AppModel();
@Override
public void initialize(URL location, ResourceBundle resources) {
text.textProperty().bindBidirectional(model.textProperty());
model.setText("Hello JavaFX.");
}
public void click(ActionEvent event) {
model.setText(appService.visitGoogle());
}
}

运行JavaFX应用,当点击第一个Go按钮之后,再点击其他按钮界面就会卡住,出现“未响应”的现象。

2. UI线程、业务线程分离

前面有提到,出现“未响应”这种情况是因为在UI线程中处理一些耗时的业务,当UI线程在处理耗时的业务时,UI就会卡住。

所以如果能将UI线程和业务线程分开来,这样就能解决界面卡住的问题了。

改造一下AppUI,将调用visitGoogle的代码放到新线程去执行。

public void click(ActionEvent event) {
new Thread(() -> model.setText(appService.visitGoogle())).start();
}

这里直接采用new的方式创建线程,实际应用中最好是使用线程池。

虽然将业务代码放在新线程中处理解决了界面卡住的问题,但是上面的代码中,通过model.setText()来改变标签(Label)的文字。

实际上会发现程序运行后会出现异常,一旦我们在非UI线程中尝试改变UI效果,程序就会抛出下面的异常。

界面卡住的问题虽然解决了,但又出现了新的问题。

3. 在UI线程更新UI

在UI线程处理业务会导致界面卡住,在业务线程更新UI会出现异常,为了能在业务线程中更新UI,JavaFX为开发者提供了一个Platform类。

只需要在业务线程中,将更新UI的代码放在这个类的runLater方法中执行即可。

下面再次改造AppUI

public void click(ActionEvent event) {
new Thread(() -> {
String title = appService.visitGoogle();
Platform.runLater(() -> model.setText(title));
}).start();
}

这里还是将业务代码放在新线程中执行,但是涉及UI更新的代码model.setText()则放在Platform.runLater()里面执行。

这样,就解决了界面卡住以及非UI线程更新UI出现异常的问题了。

通过改造,虽然请求Google超时了,但是UI并没有卡住,同时界面也得到了更新。

所以特别注意,在开发过程中应该尽量避免:

  1. 在UI线程中处理业务
  2. 在业务线程中更新UI

=========================================================

源码可关注公众号 “HiIT青年” 发送 “javafx-thread” 获取。



关注公众号,阅读更多文章。

最新文章

  1. CF 439C Devu and Partitioning of the Array
  2. [译]Mongoose指南 - Plugin
  3. 【亲述】Uber容错设计与多机房容灾方案 - 高可用架构系列
  4. A Study of WebRTC Security
  5. eclipse将编辑栏一分为二
  6. jQuery实现Checkbox中项目开发全选全不选的使用
  7. EPANET能做什么,不能做什么
  8. 【转】 IOS 项目配置--构建输出DIR
  9. 使用「max-height」实现自适应高度
  10. EIGRP默认路由分发的四种方法
  11. 【转】adobe acrobat pro修改pdf文字
  12. UDE-00008 ORA-31626 ORA-06512 ORA-25254
  13. webpack,配置,上手,例子
  14. An Overview of End-to-End Exactly-Once Processing in Apache Flink (with Apache Kafka, too!)
  15. 使用Keras训练大规模数据集
  16. java基础 二进制补码
  17. Centos7下ups监控apcupsd的使用
  18. 安装配置Glusterfs
  19. [原] ubuntu 13.10 安装 winqq2013
  20. 硬链接(hard link)和符号连接(symbolic link)

热门文章

  1. CSS数据样式
  2. 深入理解Java虚拟机--个人总结(持续更新)
  3. jdk安装无反应解决方法
  4. CISSP 考试经验分享
  5. mdk/iar汇编区别
  6. 如何将 Bitbucket 的 pull request 签出到本地 review
  7. [redis] -- 集群篇
  8. Ribbon负载均衡接口
  9. 借鉴一个比较标准的后端RESTful API
  10. 一文了解JDK12 13 14 GC调优秘籍-附PDF下载