如何制作自定义的Java/JavaFX控制台?

4
需要制作自定义控制台。我有以下代码:
public class Console extends OutputStream{

private Console(ResourceBundle resourceBundle) throws IOException {
    FXMLLoader loader = new FXMLLoader(this.getClass().getResource("Console.fxml"), resourceBundle);
    controller = loader.getController();
    Scene scene = new Scene((Parent) loader.load());
    stage = new Stage();
    stage.setScene(scene);
    show();
}

@Override
public void write(int b) throws IOException {
    controller.append(b);
}

public static Console getInstance(ResourceBundle resourceBundle) {
    if (console == null) {
        try {
            console = new Console(resourceBundle);
        } catch (IOException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }
    }
    return console;
}

public void show() {
    stage.show();
}

private static Console console = null;
private ConsoleController controller;
private Stage stage;
}

控制器文件:
public class ConsoleController implements Initializable {

@Override
public void initialize(URL url, ResourceBundle resourceBundle) {
}

@FXML
public void append(int i) {
    textArea.appendText(String.valueOf((char) i));
}

@FXML
private TextArea textArea;
}

而且所有这些都是从“start()”调用的:

    @Override
public void start(Stage primaryStage) throws IOException {
    ...
    System.setErr(new PrintStream(Console.getInstance(rb)));
}

在异常期间,Console.fxmltextArea没有输出任何内容,但是抛出了以下异常:

Exception in thread "JavaFX Application Thread" Exception: java.lang.NullPointerException thrown from the UncaughtExceptionHandler in thread "JavaFX Application Thread"

我做错了什么?

------编辑--------

在理解需要使用多线程之后,我有以下代码:
public class Console{
private Console(ResourceBundle resourceBundle) throws IOException {
    FXMLLoader loader = new FXMLLoader(this.getClass().getResource("Console.fxml"), resourceBundle);
    Parent root = (Parent) loader.load();
    controller = loader.getController();
    Scene scene = new Scene(root);
    stage = new Stage();
    stage.setScene(scene);
    show();

    if (errorOutputThread != null) {
        errorOutputThread.interrupt();
        try {
            errorOutputThread.join();
        } catch (InterruptedException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }
        errorOutputThread = null;
    }

    if (outOutputThread != null) {
        outOutputThread.interrupt();
        try {
            outOutputThread.join();
        } catch (InterruptedException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }
        outOutputThread = null;
    }

    System.err.flush();
    System.out.flush();

    outPipedInputStream = new PipedInputStream();
    outPipedOutputStream = new PipedOutputStream(outPipedInputStream);
    System.setOut(new PrintStream(outPipedOutputStream));

    errorPipedInputStream = new PipedInputStream();
    errorPipedOutputStream = new PipedOutputStream(errorPipedInputStream);
    System.setErr(new PrintStream(errorPipedOutputStream));

    outOutputThread = new Thread(new ConsoleStream(outPipedInputStream, "OUT"));
    outOutputThread.setDaemon(true);
    outOutputThread.start();

    errorOutputThread = new Thread(new ConsoleStream(errorPipedInputStream, "ERROR"));
    errorOutputThread.setDaemon(true);
    errorOutputThread.start();

    controller.appendText("Start Console");

}

public static Console getInstance(ResourceBundle resourceBundle) {
    if (console == null) {
        try {
            console = new Console(resourceBundle);
        } catch (IOException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }
    }
    return console;
}

public void show() {
    stage.show();
}

private class ConsoleStream implements Runnable {
    private ConsoleStream(InputStream in, String type) {
        inputStream = in;
        this.type = type;
    }

    public void run() {
        try {
            InputStreamReader is = new InputStreamReader(inputStream);
            BufferedReader br = new BufferedReader(is);
            String read = null;
            read = br.readLine();
            while(read != null) {
                controller.appendText(read + "\n");
                read = br.readLine();
            }
        } catch (IOException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }
        controller.appendText("Thread" + type + "started");

    }

    private final InputStream inputStream;
    private String type;
}

private static Console console = null;
private ConsoleController controller;
private Stage stage;
private PrintStream printStream;
private PipedOutputStream customPipedOutputStream;
private PipedOutputStream errorPipedOutputStream;
private PipedOutputStream outPipedOutputStream;
private PipedInputStream customPipedInputStream;
private PipedInputStream errorPipedInputStream;
private PipedInputStream outPipedInputStream;
private Thread customOutputThread;
private Thread outOutputThread;
private Thread errorOutputThread;
}

但是结果只有:“启动控制台”,没有controller.appendText("Thread" + type + "started");的结果,所以似乎这些线程没有启动。但是为什么呢?


某些东西为空(我不知道是什么)。Java 8 允许您为 JavaFX 应用程序线程设置未捕获的异常处理程序。也许,如果您在 Java 8 中测试应用程序并设置了未捕获的异常处理程序,您将能够获得更多信息(例如堆栈跟踪),这可能有助于调试问题。 - jewelsea
你可能也会对我为JavaFX迷你IDE项目创建的系统错误和系统输出控制台记录器感兴趣。这可能不是处理此问题最有效的方法,但对我来说可行。 - jewelsea
@jewelsea:我使用了你的日志记录器代码,并理解了我需要为控制台创建单独的线程。现在我遇到了另一个问题 - 这些线程不想启动。 - Eugene
1个回答

2

我认为你需要在获取控制器实例之前加载表单

FXMLLoader loader = new FXMLLoader(this.getClass().getResource("Console.fxml"),resourceBundle);
Parent p = (Parent) loader.load()
controller = loader.getController();
Scene scene = new Scene(p);

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