java - 从可运行线程追加时,JTextArea为空白-LMLPHP

我是Swing的新手,正在开发桌面应用程序。该应用程序的一部分具有运行.exe的后台进程,然后将终端的输出读取到动态创建的JtextArea中。我将此输出附加到JtextArea(线程安全)。我可以在我的PrintLn中看到输出,但不能在JTextArea上看到。实际上,JTextRea已冻结且为空白。我怀疑与重新验证和重新绘制有关,但是我只是想知道将它们放置在哪里?这是完成这项工作的代码。欢迎任何帮助或建议。

progressWindow = new JFrame("Please wait ...");

                FlowLayout defFileFlow = new FlowLayout();
               progressWindow.setLayout(defFileFlow);
                defFileFlow.setAlignment(FlowLayout.TRAILING);
                progressWindow.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                progressWindow.setSize(550,680);

                progressPane = new JTextArea(30, 80);
                DefaultCaret caret = (DefaultCaret)progressPane.getCaret();
                caret.setUpdatePolicy(DefaultCaret.OUT_BOTTOM);

               progressPane.setLayout(new BorderLayout(500, 500));
               progressPane.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));

                progressPane.setFont(new Font("Monospaced", 0, 12));
                progressPane.setText("Please wait while we crunch some numbers .." + "\n");
                JScrollPane scroller = new JScrollPane(progressPane, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS, JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS);

               progressWindow.add(scroller);
               scroller.setBounds(0, 0, 500, 500);

                JButton cancelButton = new JButton("Cancel Analysis");
                progressWindow.add(cancelButton);
                progressWindow.setComponentOrientation(ComponentOrientation.UNKNOWN);
                progressWindow.setVisible(true);
                progressWindow.setAlwaysOnTop(true);

                Dimension dim = Toolkit.getDefaultToolkit().getScreenSize();
                progressWindow.setLocation(dim.width/2-progressWindow.getSize().width/2, dim.height/2-progressWindow.getSize().height/2);


            try {
                   copyExecutable(defFilePath, selectedModel);
                   Process p=Runtime.getRuntime().exec("cmd /c dir && cd " + defFilePath + " && dir && "
                            + defFileName);
                   Thread runCMD = new Thread(new Runnable(){
                       public void run() {
                           System.out.println("Inside the thread");
                           try{
                                InputStreamReader isr = new InputStreamReader(p.getInputStream());
                                BufferedReader br = new BufferedReader(isr);
                                String line=null;  // UI magic should run in here @adityapona
                                while ( (line = br.readLine()) != null){
                                    System.out.println("MIXWILD:" + line);
                                    progressPane.append("MIXWILD:" + line + "\n");

progressPane.setCaretPosition(progressPane.getDocument().getLength());
                                }
                                } catch (IOException ioe)
                                  {
                                    ioe.printStackTrace();
                                  }
                       }
                   });
                  runCMD.start();


重要
当线程运行并成功返回退出值0时,窗口成功关闭。

最佳答案

Process p=Runtime.getRuntime().exec("...);
Thread runCMD = new Thread(new Runnable(){


该进程正在Event Dispatch Thread (EDT)上运行,这会阻止GUI响应事件并重新绘制自身。

进程本身应该从线程内部启动。 (即,尝试仅将Process语句从Thread外部移到run()方法内部,以便Process在单独的Thread上执行,而不是在EDT Thread上执行。)

阅读Swing教程中有关Swing中的并发性的部分,以获取更多信息。您可能要考虑使用为这种类型的处理设计的SwingWorker。它允许您调用长期运行的任务,并在结果可用时“发布”结果。这是比仅使用线程更好的解决方案,因为“发布”过程将确保在EDT上执行“附加”。

编辑:

progressPane.setLayout(new BorderLayout(500, 500));


为什么要尝试为文本区域提供布局管理器?怀疑会引起问题,但是绝对不需要并且不应该使用它。

progressWindow.setComponentOrientation(ComponentOrientation.UNKNOWN);


无需玩定向游戏。

scroller.setBounds(0, 0, 500, 500);


不要玩极限游戏。布局管理器的工作是设置组件的大小/位置。

10-07 12:43
查看更多