2013-03-05 94 views
0

我正在开发一个java应用程序,使用bufferedInputStream保存大文件。 我已经在JDialog中放置了一个进度条,它指示上传文件的百分比,并且每n秒增加一次。 问题是应用程序无限期地等待关闭对话框,所以它永远不会退出。 任何人都可以提供帮助吗?Java上传百分比进度条线程

这里是主应用程序的代码片段:

JDialog dialog = new JDialog(Main.getMainFrame(), true); 
        ProgressBarJPanel progressBarJPanel = new ProgressBarJPanel(); 
        dialog.setContentPane(progressBarJPanel); 
        dialog.pack(); 
        dialog.setVisible(true); 
        while ((val = bufferedInputStream.read()) != -1) 
        { 
         fileOutputStream.write(val); 
        } 
        progressBarJPanel.end(); 
        dialog.setVisible(false); 

和牵连类

import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 

public class ProgressBarJPanel extends JPanel 
     implements ActionListener 
{ 
    private JProgressBar progressBar; 
    private Timer timer; 
    public Thread updateThread; 
    public final static int ONE_SECOND = 1000; 
    private JTextArea taskOutput; 
    private String newline = "\n"; 
    int timeNow= 0; 
    int progress = 0; 

    public ProgressBarJPanel() 
    { 
     super(new BorderLayout()); 

     progressBar = new JProgressBar(0, 100); 
     progressBar.setValue(0); 
     progressBar.setStringPainted(true); 
     taskOutput = new JTextArea(5, 20); 
     taskOutput.setMargin(new Insets(5,5,5,5)); 
     taskOutput.setEditable(false); 
     taskOutput.setCursor(null); 

     JPanel panel = new JPanel(); 
     panel.add(progressBar); 

     add(panel, BorderLayout.PAGE_START); 
     add(new JScrollPane(taskOutput), BorderLayout.CENTER); 
     setBorder(BorderFactory.createEmptyBorder(20, 20, 20, 20)); 

     //Create a timer. 
     timer = new Timer(ONE_SECOND, new ActionListener() { 
      @Override 
      public void actionPerformed(ActionEvent evt) { 
       progressBar.setValue(progress); 
       progress+=10; 
       String s = "now at "+progress+"%"; 
       if (s != null) { 
        taskOutput.append(s + newline); 
        taskOutput.setCaretPosition(
          taskOutput.getDocument().getLength()); 
       } 
      } 
     }); 

    } 

    public void end() 
    { 
     timer.stop(); 
    } 

    public void startProgress() 
    { 
     timer.start(); 
    } 

    /** 
    * Create the GUI and show it. For thread safety, this method should be 
    * invoked from the event-dispatching thread. 
    */ 
    private static void createAndShowGUI() 
    { 
     //Make sure we have nice window decorations. 
     JFrame.setDefaultLookAndFeelDecorated(true); 

     //Create and set up the window. 
     JFrame frame = new JFrame("ProgressBarDemo"); 
     frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); 

     //Create and set up the content pane. 
     JComponent newContentPane = new ProgressBarJPanel(); 
     newContentPane.setOpaque(true); //content panes must be opaque 
     frame.setContentPane(newContentPane); 

     //Display the window. 
     frame.pack(); 
     frame.setVisible(true); 
    } 

    public static void main(String[] args) 
    { 
       createAndShowGUI(); 
    } 

    @Override 
    public void actionPerformed(ActionEvent e) 
    { 
     throw new UnsupportedOperationException("Not supported yet."); 
    } 
} 

- 编辑

这里是基于ben75建议解决方案:

JDialog dialog = new JDialog(Main.getMainFrame(), true); 
ProgressBarJPanel progressBarJPanel = new ProgressBarJPanel(); 
dialog.setContentPane(progressBarJPanel); 
Runnable r = new Runnable(){ 
    public void run(){ 
      SwingUtilities.invokeLater(new Runnable() 
      { 
        @Override 
        public void run() 
        { 
         progressBarJPanel.startProgress(); 
         dialog.pack(); 
         dialog.setVisible(true); 
        } 
      }); 

      //this is the long running job 
      while ((val = bufferedInputStream.read()) != -1) 
      { 
       fileOutputStream.write(val); 
      } 

      //here is the callback to UI thread 
      SwingUtilities.invokeLater(new Runnable(){ 
       public void run(){ 
        progressBarJPanel.end(); 
        dialog.setVisible(false); 
       } 
      } 
    }; 
Thread t = new Thread(r); 
t.start(); 
+1

一个忠告:使用不同的线程耗时的工作比一个处理的UI ...... – ppeterka 2013-03-05 08:59:39

+0

如何在两个沟通?同样在上面的Netbeans警告我说updateThread.start();不应该在构造函数里面,我怎么把它放在外面? – dendini 2013-03-05 09:02:55

+0

查看此示例(http://stackoverflow.com/questions/15199091/progress-bar-java/15199220#15199220) – MadProgrammer 2013-03-05 09:56:36

回答

3

你必须在另一个线程中完成时间消耗工作(即非UI线程)并在作业回调结束时回调UI线程以关闭对话框。

或多或少如何能够编码:

JDialog dialog = new JDialog(Main.getMainFrame(), true); 
ProgressBarJPanel progressBarJPanel = new ProgressBarJPanel(); 
dialog.setContentPane(progressBarJPanel); 
dialog.pack(); 
dialog.setVisible(true); 
Runnable r = new Runnable(){ 
     public void run(){ 
       //this is the long running job 
       while ((val = bufferedInputStream.read()) != -1) 
       { 
        fileOutputStream.write(val); 
       } 
       //here is the callback to UI thread 
       SwingUtilities.invokeLater(new Runnable(){ 
        public void run(){ 
         progressBarJPanel.end(); 
         dialog.setVisible(false); 
        } 
       } 
     }; 
Thread t = new Thread(r); 
t.start(); 
+0

该进程仍然挂起,因为progressJBar是在Runnable之外创建的,创建时它启动了一个线程以每秒更新其进展。您的updateThread中的 – dendini 2013-03-05 09:47:23

+0

:而不是直接更新progressBar:在UI线程的回调中执行(使用'SwingUtilities.invokeLater(...)',如我的文章所示)。 – ben75 2013-03-05 09:52:58

+0

我更新了我的问题根据您的解决方案,它的工作原理,但我仍然不确定线程​​t实际上是否总是会停止... – dendini 2013-03-05 10:17:02

3

这将是一个SwingWorker一份理想的工作 - 做上传的doInBackground,并将它每隔一段时间调用setProgress与最新进展号码。使用PropertyChangeListener更新进度栏并关闭done中的对话框,两者都保证在事件处理线程上运行。

在上面链接的javadoc中,有一个非常接近您需要的示例。

0

把它全部带走,并使用javax.swing.ProgressMonitorInputStream.

+0

怎么样?这不仅仅是一个回答而是一个评论。 – 2014-05-21 16:27:17