0

Я разрабатываю приложение java, где я сохраняю большие файлы с помощью bufferedInputStream. Я поставил индикатор прогресса внутри JDialog, который указывает процент загруженных файлов и который увеличивается каждые n секунд. Проблема заключается в том, что приложение ждет неопределенное время для закрытия диалогового окна и поэтому никогда не выходит. Кто-нибудь может помочь?Java uploading percent progressbar thread

здесь главное приложение сниппет:

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."); 
    } 
} 

- EDIT

Вот это решение, основанное на 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

+0

Как два общаетесь? Также в приведенном выше Netbeans предупреждает меня, что updateThread.start(); не должно быть внутри конструктора, как я могу поставить его вне? – dendini

+0

Проверьте [этот пример] (http://stackoverflow.com/questions/15199091/progress-bar-java/15199220#15199220) – MadProgrammer

ответ

3

Вы должен выполнять время, выполняемое в другом потоке (т. non 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 и при создании запускает поток чтобы обновить свой прогресс каждую секунду. – dendini

+0

в вашем updateThread: вместо обновления progressBar напрямую: сделайте это в обратном вызове для потока пользовательского интерфейса (используйте 'SwingUtilities.invokeLater (...)', как показано в моем сообщении). – ben75

+0

Я обновил свой вопрос на основе вашего решения, он работает, но я все еще не уверен, будет ли поток t на самом деле всегда останавливаться ... – dendini

3

Это будет идеальная работа для SwingWorker - сделать загрузку в doInBackground, и иметь его называют setProgress с обновленным номером прогресса каждый так часто. Обновите индикатор выполнения с помощью PropertyChangeListener и закройте диалоговое окно done, оба из которых гарантированно будут работать в потоке обработки событий.

Существует пример, очень близкий к тому, что вам нужно в javadoc, с которым я связан выше.

0

Бросьте все это прочь и использовать javax.swing.ProgressMonitorInputStream.

+0

как? Это скорее комментарий, чем ответ. –