我是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);
不要玩极限游戏。布局管理器的工作是设置组件的大小/位置。