4

我无法回答的关于 JavaFX 的新手问题,尽管我知道它必须非常简单,并且在我看过的任何地方都找不到任何资源(教程、许多 Oracle 在线文档、文章、以及知名 JavaFX 博主等)

我正在开发一个命令行(脚本)运行应用程序,并且我已经成功地从脚本中获得了输出(通过 ProcessBuilder),我可以以持续的方式显示,因为命令行上发生了一些事情。也就是说,我可以System.out.println(line);整天做,在控制台中显示输出,它只是从正在运行的“myProcess”返回的输入流中返回输出,创建如下:

BufferedReader bri = new BufferedReader(new InputStreamReader(myProcess.getInputStream()))

所以我能够看到脚本返回的所有输出。

我想设置一个 JavaFX TextArea 或 ScrollPane 或者,不确定是什么,来显示这个输出文本(有很多,比如几千行)作为脚本中正在发生的事情的持续“进展”,当它发生的时候。我有一个场景,我有按钮并从该场景获取输入以启动脚本运行,但现在我想显示单击“运行此脚本”按钮的结果,可以这么说。

我假设我需要按照此处所述创建一个 TextArea,或者TextBuilder可能对开始制作它很有用。没有把握。

我需要一些帮助来设置绑定或自动滚动/自动更新部分。

有人可以给我一个开始的地方,用 JavaFX 做这个吗?我宁愿不使用 Swing。

(我正在使用 JavaFX 2.2、JDK 1.7u7 和所有最新的东西,是的,这是一个 FXML 应用程序——所以最好这样做。)

更新: Sergey Grinev 的回答在绑定部分非常有帮助。但是这里有一些关于我要求“如何设置帮助”时的更多详细信息——基本上,我需要将控制权返回到主场景以允许用户取消脚本,或者以其他方式监控这是怎么回事。所以我想“生成”运行该脚本的进程(也就是说,有某种“自由运行的进程”),但仍然从中获得输出。(在我最初的问题中,我不是很清楚。)

我在这里使用的技术(见下文)是对进程执行 waitFor ,但这当然意味着在脚本执行时对话框/场景是“挂起”的。我想重新获得控制权,但是如何将“p”(进程)传递给其他控制器部分(或者,简单地启动传递参数的其他进程以启动脚本并让它启动脚本) 然后通过绑定 Sergey Grinev 提到的方式进行自动更新——而不“挂起”场景/窗口?另外:如果用户请求,我可以“停止”这个其他进程吗?

这是我当前的代码(“等待”脚本——运行需要 20-40 分钟!——完成;这不是我想要的,我希望将控制权返回给用户):

public class MyController implements Initializable {  
  @FXML
  private void handleRunScript(ActionEvent event) throws IOException {

    ProcessBuilder pb = new ProcessBuilder("myscript.sh", "arg1", "arg2", ...);

    Process p = pb.start();
    try {
      BufferedReader bri = new BufferedReader
        (new InputStreamReader(p.getInputStream()));
      String line;

      while ((line = bri.readLine()) != null) {
        System.out.println(line);
        textAreaRight.setText(line);
      }
      bri.close();
      p.waitFor();
    }
    catch (Exception err) {
      err.printStackTrace();
    }    
  }

  @FXML
  private void handleCancel(ActionEvent event) {
    doSomethingDifferent();
  }
}
4

1 回答 1

5
  1. 要记录字符串,您可以使用 TextArea
  2. 为了使其异步,您需要为输出阅读器创建一个单独的线程。

public class DoTextAreaLog extends Application {

    TextArea log = new TextArea();
    Process p;

    @Override
    public void start(Stage stage) {
        try {
            ProcessBuilder pb = new ProcessBuilder("ping", "stackoverflow.com", "-n", "100");

            p = pb.start();

            // this thread will read from process without blocking an application
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        //try-with-resources from jdk7, change it back if you use older jdk
                        try (BufferedReader bri = new BufferedReader(new InputStreamReader(p.getInputStream()))) {
                            String line;

                            while ((line = bri.readLine()) != null) {
                                log(line);
                            }
                        }
                    } catch (IOException ex) {
                        ex.printStackTrace();
                    }
                }
            }).start();

            stage.setScene(new Scene(new Group(log), 400, 300));
            stage.show();
        } catch (IOException ex) {
            ex.printStackTrace();
        }

    }

    @Override
    public void stop() throws Exception {
        super.stop();
        // this called on fx app close, you may call it in user action handler
        if (p!=null ) {
            p.destroy();
        }
    }

    private void log(final String st) {
        // we can access fx objects only from fx thread
        // so we need to wrap log access into Platform#runLater
        Platform.runLater(new Runnable() {

            @Override
            public void run() {
                log.setText(st + "\n" + log.getText());
            }
        });
    }

    public static void main(String[] args) {
        launch();
    }
}
于 2012-09-07T16:35:56.673 回答