[英]How to wait for a JFrame thread to finish from inside a JavaFX application?
我正在尝试从JavaFX应用程序内部创建JFrame窗口。 我正在使用Oracle Java 8,并且能够执行Linux,Windows而不是Mac OS中所需的操作。 似乎是JVM实现问题。 这是我的要求:
我在下面创建了示例应用程序。 应用程序在Linux和Windows下运行良好,但是在Mac OS中,当我尝试等待JFrame线程完成应用程序冻结时,未显示JFrame。 SwingUtilities.invokeLater不会冻结,但不会等待JFrame线程完成。 由于我需要等待JFrame线程,invokeLater并不是我的选择。
JFrameCreator.java
public class JFrameCreator implements Runnable {
@Override
public void run() {
System.out.println("Creating JFrame");
JFrame frame=new JFrame();
frame.setDefaultCloseOperation(JFrame.HIDE_ON_CLOSE);
frame.setSize(300,300);
frame.setVisible(true);
for (int i = 0; i < 100; i++) {
frame.getContentPane().getGraphics().drawOval(150-i/2,150-i/2,i,i);
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
sample.fxml
<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="100.0" prefWidth="200.0" xmlns="http://javafx.com/javafx/8.0.111" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
<children>
<Button layoutX="51.0" layoutY="10.0" mnemonicParsing="false" onAction="#createJFrame" text="New JFrame" />
<RadioButton fx:id="cbThread" layoutX="10.0" layoutY="40.0" mnemonicParsing="false" selected="true" text="Thread">
<toggleGroup>
<ToggleGroup fx:id="tg" />
</toggleGroup>
</RadioButton>
<RadioButton fx:id="cbInvokeWait" layoutX="10.0" layoutY="60.0" mnemonicParsing="false" text="invokeAndWait" toggleGroup="$tg" />
<RadioButton fx:id="cbInvokeLater" layoutX="10.0" layoutY="80.0" mnemonicParsing="false" text="invokeLater" toggleGroup="$tg" />
</children>
</AnchorPane>
Controller.java
public class Controller {
@FXML RadioButton cbThread;
@FXML RadioButton cbInvokeWait;
@FXML RadioButton cbInvokeLater;
public void createJFrame(javafx.event.ActionEvent actionEvent) {
Runnable r=new JFrameCreator();
try {
if (cbThread.isSelected()) {
Thread t=new Thread(new JFrameCreator());
t.start();
t.join();
}
else if(cbInvokeWait.isSelected())
SwingUtilities.invokeAndWait(r);
else
SwingUtilities.invokeLater(r);
}
catch (InterruptedException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}
Main.java
public class Main extends Application {
@Override
public void start(Stage primaryStage) throws Exception{
Parent root = FXMLLoader.load(getClass().getResource("sample.fxml"));
primaryStage.setTitle("Swing in JavaFX");
primaryStage.setScene(new Scene(root, 200, 100));
primaryStage.show();
}
public static void main(String[] args) {
launch(args);
}
}
我还尝试了Thread.isAlive()方法的while循环,但它的行为也类似于Thread.join()。 如何达到我的要求?
您可以使用同步类之一,例如CountDownLatch
。 将其传递给您的Runnable
并在run()
方法的末尾调用latch.countDown()
。 同时,您的JavaFX线程将暂停在latch.await()
。 但是,这将冻结JavaFX UI,所以这可能不是您想要的。
另一种选择是使用回调。 就像是:
public class JFrameCreator implements Runnable {
private final Runnable onFinished;
public JFrameCreator(Runnable onFinished) {
this.onFinished = onFinished;
}
@Override
public void run() {
// do your work...
Platform.runLater(onFinished); // to communicate to JavaFX Application
// Thread that JFrameCreator is complete
}
}
第二个选项不会导致JavaFX线程等待,但它确实允许您在适当的时间做出反应。 您可以将onFinished
的类型更改为onFinished
的任何内容。 例如Consumer
如果您想传递回调某些对象。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.