Swing JFrame导致JavaFX应用在OS X上崩溃

3
我正在开发一个复杂的JavaFX项目,其中包括Swing JFrames。所有功能都正常,但在某些情况下JFrames却无法关闭,导致整个虚拟机挂起。
我将问题简化为最小工作示例,此示例在Windows上似乎可以正常运行(请确认是否能在您的计算机上运行),但在OS X上会可靠地崩溃。我使用的是Java 8u25(最新稳定版)和8u40预览版(最新版本)-没有区别。
如何重现:将程序保存为“JavaFXWithJFrames.java”,编译并运行。现在有3个Swing JFrames和1个带有3个按钮的JavaFX窗口。单击按钮应分别关闭1个窗口。这在Windows上有效吗?但完全锁定了OS X(以及其他操作系统?)
您能否重现这个问题?是/否,使用的机器/操作系统/JRE是什么?我做错了什么-线程问题?非常感谢您的帮助。 pastebin上的代码:http://pastebin.com/tUrdNfCw# - 请另存为“JavaFXWithJFrames.java”进行编译!
/*
 * To change this license header, choose License Headers in Project Properties.
 * To change this template file, choose Tools | Templates
 * and open the template in the editor.
 */
//package javafxwithjframes;

import javafx.application.Application;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.HBox;
import javafx.stage.Stage;
import javafx.stage.WindowEvent;
import javax.swing.JFrame;

/**
 * When starting, a JavaFX window and 3 Swing JFrame-windows appear.
 * The JavaFX window features 3 buttons that each are supposed to close one of the JFrames.
 * The program, however, freezes after clicking a button. What am I doing wrong?
 * It seems to work on Windows, but crash on OS X 10.10.1, using Java 8u25 (latest release) and Java 8u40 Dec 31 preview-build.
 *
 * @author a desperate developer
 */
public class JavaFXWithJFrames extends Application {

JFrame jframe1, jframe2, jframe3;

@Override
public void start(Stage primaryStage) {

    // Create 3 JFrames
    jframe1 = new JFrame("JFrame 1");
    jframe1.setBounds(50, 50, 200, 150);
    jframe1.setVisible(true);

    jframe2 = new JFrame("JFrame 2");
    jframe2.setBounds(275, 50, 200, 150);
    jframe2.setVisible(true);

    jframe3 = new JFrame("JFrame 3");
    jframe3.setBounds(500, 50, 200, 150);
    jframe3.setVisible(true);

    // Create 3 buttons that close each one JFrame
    // Button 1
    Button closeButton1 = new Button();
    closeButton1.setText("Close JFrame 1");
    closeButton1.setOnAction(new EventHandler<ActionEvent>() {
        @Override
        public void handle(ActionEvent event) {
            jframe1.setVisible(false);
            System.out.print("Closing JFrame 1...");
            jframe1.dispose();
        }
    });

    // Button 2
    Button closeButton2 = new Button();
    closeButton2.setText("Close JFrame 2");
    closeButton2.setOnAction(new EventHandler<ActionEvent>() {
        @Override
        public void handle(ActionEvent event) {
            jframe2.setVisible(false);
            System.out.print("Closing JFrame 2...");
            jframe2.dispose();
        }
    });

    // Button 3
    Button closeButton3 = new Button();
    closeButton3.setText("Close JFrame 3");
    closeButton3.setOnAction(new EventHandler<ActionEvent>() {
        @Override
        public void handle(ActionEvent event) {
            jframe3.setVisible(false);
            System.out.print("Closing JFrame 3...");
            jframe3.dispose();
        }
    });

    // Setting up main window
    HBox rootBox = new HBox();
    rootBox.getChildren().addAll(closeButton1, closeButton2, closeButton3);
    Scene scene = new Scene(rootBox, 400, 250);
    primaryStage.setOnCloseRequest(new EventHandler<WindowEvent>() {
        @Override
        public void handle(WindowEvent event) {
            System.exit(0);
        }
    });
    primaryStage.setTitle("JavaFX with JFrames");
    primaryStage.setScene(scene);
    primaryStage.show();
}

/**
 * @param args the command line arguments
 */
public static void main(String[] args) {
    launch(args);
}

}

对我来说它很好用(Windows 8.1,64位,JDK8u25 64位)。 - José Pereda
1个回答

6
您在这里有一个线程问题:您正在FX应用程序线程上执行所有操作,而Swing工作应该在AWT事件分派线程上进行。
您可以使用SwingUtilities.invokeLater(...); 将代码调度到AWT事件处理线程上运行。
所以您的代码应该如下所示:
@Override
public void start(Stage primaryStage) {

    SwingUtilities.invokeLater( () -> {
        // Create 3 JFrames
        jframe1 = new JFrame("JFrame 1");
        jframe1.setBounds(50, 50, 200, 150);
        jframe1.setVisible(true);

        jframe2 = new JFrame("JFrame 2");
        jframe2.setBounds(275, 50, 200, 150);
        jframe2.setVisible(true);

        jframe3 = new JFrame("JFrame 3");
        jframe3.setBounds(500, 50, 200, 150);
        jframe3.setVisible(true);
    });

    // Create 3 buttons that close each one JFrame
    // Button 1
    Button closeButton1 = new Button();
    closeButton1.setText("Close JFrame 1");
    closeButton1.setOnAction(event -> {
        SwingUtilities.invokeLater(() -> {
            jframe1.setVisible(false);
            System.out.print("Closing JFrame 1...");
            jframe1.dispose();
        });
    });

    // similarly for other button handlers...

}

我将您的事件处理程序更改为lambda表达式,因为嵌套的内部类太丑了,并且您说您正在使用JDK 8...
您也可以尝试按原样运行应用程序,但使用实验性系统属性
-Djavafx.embed.singleThread=true 这会在同一线程上运行两个UI工具包。由于(据我所知)这仍然是实验性的,我建议按照上面显示的“正确”线程安排代码。

哇,谢谢!这就解决了问题!很高兴我至少朝着正确的方向努力 :) - underkuerbis

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接