java - JavaFX Platform.runLater 的使用和从不同线程访问 UI

标签 java multithreading javafx-2 event-dispatch-thread

我有几个关于 Platform.runLater 的问题.我有一个 JavaFX 应用程序类。在这个类中,我运行一个线程(该线程从网络套接字读取数据)。

现在当我创建一个新的 Stage在线程内,系统抛出一个 execption(JavaFX 事件调度线程和我的网络读取线程不一样)——我理解这种行为。

但另一方面,我将网络阅读器中的文本附加到现有的 TextArea或在 ListView<String> 中添加/删除一些项目- 这不会引发异常 - 为什么?我认为 JavaFX 是单线程的(ui 库部分)。这与 Swing 中的一样吗:有时它可以工作,有时却只有垃圾(因为 EDT)?

我的问题:

  • JavaFX 事件调度程序线程何时抛出异常,何时不抛出异常?
  • 关于这个有什么好的文档吗
  • 有没有更简单(更短和更清晰)的使用方式Platform.runLaterrun()方法 ?结合try catch(或multiple catch),看起来很奇怪

我知道 Platform.runLater 的用法在线程中不是很好(设计解决方案)

最佳答案

Alexander 的回答捕获了与您的问题相关的最重要的要点。

此答案提供了一些补充信息。

When does the JavaFX event dispatcher thread throw an exception and when not ?

JavaFX 系统并不总是检查对影响 Activity 场景图的对象的访问是否适本地限制在 JavaFX 线程中。最终,确保这种线程安全是 JavaFX 应用程序程序员的责任,而不是 JavaFX 系统。在 JavaFX 中执行多线程编程时必须非常小心,否则应用程序行为可能会失败或变得不可预测。

Are the any good documents about this

尝试 JavaFX 教程:Concurrency in JavaFX

Is there an easier (shorter & cleaner) way to use Platform.runLater with a run() method?

没有。 Platform.runLater 非常简单。


顺便说一句。 . .

任务与服务

考虑使用 TaskServiceWorker 子类。这些是 FutureTask(又是 Runnable)的 JavaFX 包装器。 Workers 提供了一个 call 方法来在后台线程上运行逻辑。它们保持执行 status(针对状态更改向 JavaFX 线程发送安全回调通知)并通过 valuemessageexception 属性返回调用结果。

利用 TaskService javadoc 示例中的设计模式来简化具有以下功能的线程安全应用程序的创建:

  • 为 UI 更新异步获取数据。
  • 任务进度的定期消息更新。
  • 构建尚未连接到显示场景的节点图。
  • 通过进度条等监控进度

Worker 是对 Platform.runLater 的补充。当您在 JavaFX 应用程序线程之外执行并且希望在 JavaFX 应用程序线程上运行某些逻辑时,请使用 Platform.runLater。当您在 JavaFX 应用程序线程上运行并希望在新线程上生成一些逻辑或(尤其是)I/O 时,请使用 Worker,这样您就不会阻塞 JavaFX 应用程序线程。您永远不想在 Platform.runLaterrun 方法中执行网络 I/O,但通常希望在 Workercall 方法。

此外,TaskService 的使用与 Platform.runLater 的使用并不矛盾。例如,如果您有一个运行时间很长的 Task,您希望从中定期或在缓冲区填充时将部分结果返回到 UI,然后在task 的 call 方法就是这样做的方法。

当您没有库提供的现有线程服务,而是创建自己的线程在后台执行时,Worker 很有用。如果您有现有的线程服务,则需要使用 Platform.runLater 在 JavaFX 应用程序线程上执行逻辑。

请注意,即使您使用 Worker,您仍然需要知道自己在做什么。您仍然必须注意不要违反标准的 JavaFX 并发规则,例如从不更新 Activity 场景图中的节点(包括不更新 Activity 场景图中的节点绑定(bind)到的值 - 例如 items 支持的可观察列表一个 ListView )。

关于java - JavaFX Platform.runLater 的使用和从不同线程访问 UI,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15160410/

相关文章:

Java JPanel 清除

java - 如何禁用 primefaces 中的 Accordion 面板选项卡?

java - 使用java线程的多监视器同步

javafx-2 - 如何在WebView中自动滚动到末尾?

java - 双击行时无法触发 MouseEvent

java - 为什么 JavaFX 文档与 Java 8 文档分开?

Java 按字母顺序对列表进行排序,末尾值为 "Other"

java - 如何获取用户输入的复数

multithreading - 用餐哲学家问题中的饥饿

c# - TextBox自定义ContextMenu in Style,多线程错误