转载自 JavaFX 2.1:Toolkit not initialized
一、问题
我的应用程序是基于Swing的。我想介绍JavaFX并将其配置为在辅助显示器上渲染场景。我可以使用JFrame来保存一个可以容纳JFXPanel的JFXPanel,但我想用JavaFX API实现这个目的。
子类化com.sun.glass.ui.Application并使用Application.launch(this)不是一个选项,因为调用线程将被阻止。
从Swing EDT实例化一个Stage时,我得到的错误是:
java.lang.IllegalStateException: Toolkit not initialized
问题:非平凡的Swing GUI应用程序需要运行JavaFX组件。应用程序的启动过程在启动从属服务层后初始化GUI。
解决方案
子类JavaFX Application类并在单独的线程中运行它,例如:
public class JavaFXInitializer extends Application {
@Override
public void start(Stage stage) throws Exception {
// JavaFX should be initialized
someGlobalVar.setInitialized(true);
}
}
旁注:因为Application.launch()方法接受一个Class
参数,所以必须使用全局变量来表示已初始化JavaFX环境。
替代方法:在Swing Event Dispatcher Thread中实例化JFXPanel:
final CountDownLatch latch = new CountDownLatch(1);
SwingUtilities.invokeLater(new Runnable() {
public void run() {
new JFXPanel(); // initializes JavaFX environment
latch.countDown();
}
});
latch.await();
通过使用此方法,调用线程将等待直到设置JavaFX环境。
选择您认为合适的解决方案。我选择了第二个,因为它不需要一个全局变量来表示JavaFX环境的初始化,也不会浪费一个线程。
(1)回答1
使用JavaFX的唯一方法是子类化Application或使用JFXPanel,正是因为他们准备了env和toolkit。
阻塞线程可以通过使用来解决new Thread(...)
。
虽然如果您在与Swing / AWT相同的VM中使用JavaFX,我建议使用JFXPanel,您可以在此处找到更多详细信息:将AWT与JavaFx一起使用是否可以?
(2)回答2
找到了解决方案。如果我只是在调用JavaFX Platform.runLater之前从Swing EDT创建一个JFXPanel它就可以了。我不知道这个解决方案有多可靠,如果结果不稳定,我可能会选择JFXPanel和JFrame。
public class BootJavaFX {
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
new JFXPanel(); // this will prepare JavaFX toolkit and environment
Platform.runLater(new Runnable() {
@Override
public void run() {
StageBuilder.create()
.scene(SceneBuilder.create()
.width(320)
.height(240)
.root(LabelBuilder.create()
.font(Font.font("Arial", 54))
.text("JavaFX")
.build())
.build())
.onCloseRequest(new EventHandler() {
@Override
public void handle(WindowEvent windowEvent) {
System.exit(0);
}
})
.build()
.show();
}
});
}
});
}
}
(3)回答三
我在创建用于测试javaFX tableview更新的unittests时使用了以下命令
public class testingTableView {
@BeforeClass
public static void initToolkit() throws InterruptedException
{
final CountDownLatch latch = new CountDownLatch(1);
SwingUtilities.invokeLater(() -> {
new JFXPanel(); // initializes JavaFX environment
latch.countDown();
});
if (!latch.await(5L, TimeUnit.SECONDS))
throw new ExceptionInInitializerError();
}
@Test
public void updateTableView() throws Exception {
TableView yourTable = new TableView<>();
.... do your testing stuff
}
}
尽管这篇文章与测试无关,但它帮助我让我的单元测试工作
- 没有BeforeClass initToolkit,那么在unittest中实例化TableView会产生一个缺少工具包的消息
(4)回答4
还可以通过调用以下方式显式初始化工具包: com.sun.javafx.application.PlatformImpl#startup(Runnable)
有点hacky,由于使用* Impl,但是如果你不想使用Application
或JXFPanel
出于某种原因很有用。
从这篇文章重新发布自己
(5)回答五
我检查了源代码,这是初始化它
com.sun.javafx.application.PlatformImpl.startup(()->{});
并退出它
com.sun.javafx.application.PlatformImpl.exit();