从可运行的线程追加时,JTextArea为空

时间:2017-10-24 16:11:59

标签: java swing jtextarea runnable repaint

enter image description here

我是Swing的新手,我正在开发桌面应用。部分应用程序具有运行.exe的后台进程,然后将终端的pout读入动态创建的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。

1 个答案:

答案 0 :(得分:1)

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

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

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

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

编辑:

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

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

progressWindow.setComponentOrientation(ComponentOrientation.UNKNOWN);

无需使用方向。

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

不要玩边界。布局管理器的工作是设置组件的大小/位置。