我需要找到一种在进度条上使用Swing计时器的方法。我尝试使用Thread.sleep(),但使用它时该应用程序崩溃了。有什么方法可以使用Swing计时器而不是Sleep()?
public void piiEros(int dist) { Pii pii = new Pii(); pii.setVisible(true); for(int pc = 0;100 > pc; pc++) { try { Thread.sleep(dist/100); } catch (InterruptedException ex) { Logger.getLogger(Trav.class.getName()).log(Level.SEVERE, null, ex); } pii.pg.setValue(pc); } pii.dispose(); o.Eros(); }
注意:Pii是带有进度条的课程。Dist是加载速度。Trav是该方法所在的类。Pc代表%,完成多少显示在栏上。o.Eros打开另一个图形用户界面。
(从长远来看)使用可能更容易SwingWorker。它提供了许多有用的方法来更新UI(从事件调度线程的上下文中),同时允许在后台继续执行…
SwingWorker
import java.awt.BorderLayout; import java.awt.Dimension; import java.awt.EventQueue; import java.awt.GridBagLayout; import java.beans.PropertyChangeEvent; import java.beans.PropertyChangeListener; import java.util.List; import javax.swing.JFrame; import javax.swing.JPanel; import javax.swing.JProgressBar; import javax.swing.SwingWorker; import javax.swing.UIManager; import javax.swing.UnsupportedLookAndFeelException; public class TestSwingWorker02 { public static void main(String[] args) { new TestSwingWorker02(); } public TestSwingWorker02() { EventQueue.invokeLater(new Runnable() { @Override public void run() { try { UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); } catch (ClassNotFoundException ex) { } catch (InstantiationException ex) { } catch (IllegalAccessException ex) { } catch (UnsupportedLookAndFeelException ex) { } JFrame frame = new JFrame("Test"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLayout(new BorderLayout()); frame.add(new TestPane()); frame.pack(); frame.setLocationRelativeTo(null); frame.setVisible(true); } }); } public class TestPane extends JPanel { public TestPane() { setLayout(new GridBagLayout()); JProgressBar pb = new JProgressBar(); add(pb); new ProgressWorker(pb, 40).execute(); } @Override public Dimension getPreferredSize() { return new Dimension(200, 200); } } public class ProgressWorker extends SwingWorker<Void, Integer> { private int delay; private JProgressBar pb; public ProgressWorker(JProgressBar progressBar, int delay) { this.pb = progressBar; this.delay = delay; } @Override protected void process(List<Integer> chunks) { // Back in the EDT... pb.setValue(chunks.get(chunks.size() - 1)); // only care about the last one... } @Override protected Void doInBackground() throws Exception { for (int index = 0; index < 100; index++) { publish(index); Thread.sleep(delay); } return null; } @Override protected void done() { // Back in the EDT... //pii.dispose(); //o.Eros(); } } }
将SwingWorker让你的逻辑中分离出来。在该doInBackground方法中,您可以专注于需要在EDT之外运行的那部分代码,可以将其publish更新回EDT并process分别更新。当您完成所有操作后,done您可以根据需要进行清理。
doInBackground
publish
process
done
SwingWorker还提供了进度监视功能,因此,就您而言,如果您不想这样做,则不必使用API 的publish/ process部分。这样一PropertyChangeListener来,您无需将进度条暴露在工作人员上即可将其附加到工作人员。(但我为示例做了)
PropertyChangeListener
public class ProgressWorker extends SwingWorker<Void, Integer> { private int delay; private JProgressBar pb; public ProgressWorker(JProgressBar progressBar, int delay) { this.pb = progressBar; this.delay = delay; // You can use a property change listener to monitor progress updates... addPropertyChangeListener(new PropertyChangeListener() { @Override public void propertyChange(PropertyChangeEvent evt) { if ("progress".equalsIgnoreCase(evt.getPropertyName())) { pb.setValue((Integer)evt.getNewValue()); } } }); } @Override protected Void doInBackground() throws Exception { for (int index = 0; index < 100; index++) { setProgress(index); Thread.sleep(delay); } return null; } @Override protected void done() { // Back in the EDT... //pii.dispose(); //o.Eros(); } }