多次运行SwingWorker

时间:2016-02-15 15:58:34

标签: java multithreading swing user-interface swingworker

我想生成一个随机数5次并重复6次。每次生成随机数时,我都会使用该编号和进度条更新GUI。

为了做到这一点,我想到了一个在doInBackground()方法上生成随机数的SwingWorker,并且使用publish()方法在GUI上显示该数字。我没有遇到任何问题就完成了这个问题,当我尝试这样做6次时出现问题,因为在done()方法中我在TextField上添加了一个空白区域,用于显示所有数字

如何运行SwingWorker的6倍,并在SwingWorker的done()方法完成后启动?

谢谢!

修改:代码段

public class SwingWorkerEjemplo extends SwingWorker<Void, Integer> {
private JProgressBar pBar;
private JTextField txtSalida;
private JTextArea txtArea;

public SwingWorkerEjemplo(JProgressBar pBar,JTextField txt, JTextArea txtArea) {
    this.pBar = pBar;
    addPropertyChangeListener(new PropertyChangeListener() {
        @Override
        public void propertyChange(PropertyChangeEvent evt) {
            if ("progress".equals(evt.getPropertyName())){
                pBar.setValue((Integer)evt.getNewValue());
            }
        }
    });

    this.pBar.setVisible(true);
    this.pBar.setStringPainted(true);
    this.pBar.setValue(0);
    setProgress(0);
    this.txtSalida = txt;
    this.txtArea = txtArea;
}

@Override
protected Void doInBackground() throws Exception {
    int num;
    for (int i=0;i<5;i++){
        num = ThreadLocalRandom.current().nextInt(1, 7);
        publish(num);
        setProgress(i+1);
        Thread.sleep(100);
    }
    return null;
}

@Override
protected void done() {
    try {
        txtSalida.setText(txtSalida.getText()+" ");
        Thread.sleep(0);
        txtArea.append("");
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

@Override
protected void process(List<Integer> chunks) {
    Integer valor = chunks.get(chunks.size()-1);
    txtSalida.setText(txtSalida.getText()+String.valueOf(valor));
}

}

在MainWindow.java中

SwingWorkerEjemplo swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 1");
swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 2");
swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 3");
swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 4");
swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 5");
swe = new SwingWorkerEjemplo(pBar, txtNumeros, txtSalida);
swe.execute();
System.out.println("Worker 6");

结果首先是全部 工人1 工人2 工人3 工人4 工人5 工人6 然后像xxxxxyyyyyzzzzzaaaaabbbbbcccccddddd

1 个答案:

答案 0 :(得分:3)

  

如何运行SwingWorker的6倍,并在SwingWorker的done()方法完成后启动?

你做不到。根据{{​​3}}:

  

SwingWorker只能执行一次。多次执行SwingWorker不会导致两次调用doInBackground方法。

因此,如果您需要一个延迟重复的操作,请在您的worker中使用while循环执行此操作,然后通过SwingWorker的发布/进程对将结果输出到GUI。如果你走这条路线,你就会创建一个SwingWorker<Void, Integer>(如果它需要产生int输出)。另一方面,如果你想要运行6次工作,每次响应一个事件,那么每次需要时都要创建一个新工人。

我想知道你是否只需要更简单的Swing Timer。

修改

  

我还没有看到Swing Timer的功能,所以我无法回答这个问题。

如果您需要间歇性地延迟完成某项工作,那么SwingWorker API documentation(请参阅链接)是可行的方法。当你创建一个传递2个参数时,一个表示延迟时间的int,一个ActionListener - 其actionPerformed方法将被重复调用,将粗略地延迟。因此,如果您需要每秒生成一个随机int,您将传入1000(1000毫秒= 1秒)和一个ActionListener,并在侦听器的actionPerformed方法中生成随机int并将其传递给任何需要它的人。请注意,在Swing事件线程上调用actionPerformed方法中的所有代码。

另一方面,如果需要运行一些需要很长时间才能执行的代码,例如数据库查找,从套接字读取,从文件读取或写入,那么您将使用SwingWorker因为通过这样做,您可以在后台线程中运行长段代码,并且事件线程可以安全地将信息提取回GUI。

  

快速问题,工人是指GUI吗?

不,worker = SwingWorker。

所以使用你发布的代码,如果这就是你要做的 - 用延迟更新JProgressBar,然后使用Swing Timer。另一方面,如果您希望执行长时间运行的进程并希望在进程运行时更新JProgressBar,则使用SwingWorker。请注意,如果您具有后者需求,则无需使用发布/处理方法对。 SwingWorker有一个progress属性,它是一个“绑定”属性,这意味着如果你通过调用setProgress(...)来改变它,SwingWorker将通知任何可能附加到它的PropertyChangeListener。我用这个很多

例如:

import java.awt.event.*;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.Random;
import java.util.concurrent.ExecutionException;

import javax.swing.*;

public class ProgressExampleGui {
   private JPanel mainPanel = new JPanel();
   private JProgressBar progressBar = new JProgressBar();
   private JButton pressMeBtn  = new JButton(new MyAction("Press Me", KeyEvent.VK_P, this));

   public ProgressExampleGui() {
      progressBar.setStringPainted(true);
      progressBar.setString("");

      mainPanel.add(pressMeBtn);
      mainPanel.add(progressBar);
   }

   public void setProgress(int progress) {
      progressBar.setValue(progress);
      progressBar.setString(progress + "%");
   }

   public JComponent getMainComponent() {
      return mainPanel;
   }

   public void setEnabled(boolean enabled) {
      pressMeBtn.setEnabled(enabled);
   }

   private static void createAndShowGui() {
      ProgressExampleGui progExampleGui = new ProgressExampleGui();

      JFrame frame = new JFrame("Progress Example");
      frame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
      frame.getContentPane().add(progExampleGui.getMainComponent());
      frame.pack();
      frame.setLocationByPlatform(true);
      frame.setVisible(true);
   }

   public static void main(String[] args) {
      SwingUtilities.invokeLater(new Runnable() {
         public void run() {
            createAndShowGui();
         }
      });
   }
}

@SuppressWarnings("serial")
class MyAction extends AbstractAction {
   private ProgressExampleGui gui;

   public MyAction(String name, int mnemonic, ProgressExampleGui gui) {
      super(name);
      putValue(MNEMONIC_KEY, mnemonic);
      this.gui = gui;
   }

   @Override
   public void actionPerformed(ActionEvent e) {
      AbstractButton source = (AbstractButton) e.getSource();
      gui.setProgress(0);
      source.setEnabled(false);
      MyWorker myWorker = new MyWorker();
      myWorker.addPropertyChangeListener(new WorkerPropChngListener(gui));
      myWorker.execute();
   }
}

class WorkerPropChngListener implements PropertyChangeListener {

   private ProgressExampleGui gui;

   public WorkerPropChngListener(ProgressExampleGui gui) {
      this.gui = gui;
   }

   @Override
   public void propertyChange(PropertyChangeEvent pcEvt) {
      MyWorker myWorker = (MyWorker) pcEvt.getSource();
      if ("progress".equals(pcEvt.getPropertyName())) {
         int progress = ((Integer)pcEvt.getNewValue()).intValue();
         gui.setProgress(progress);
      }

      if (SwingWorker.StateValue.DONE.equals(pcEvt.getNewValue())) {
         try {
            myWorker.get();
         } catch (InterruptedException | ExecutionException e) {
            e.printStackTrace();
         }
         gui.setEnabled(true);
      }
   }

}

class MyWorker extends SwingWorker<Void, Void> {
   private static final int MAX_INCR = 8;
   private static final long SLEEP_TIME = 200;
   private static final int MAX_VALUE = 100;
   private int value = 0;
   private Random random = new Random();

   @Override
   protected Void doInBackground() throws Exception {
      while (value < MAX_VALUE) {
         value += random.nextInt(MAX_INCR);
         value = Math.min(value, MAX_VALUE);
         Thread.sleep(SLEEP_TIME);
         setProgress(value);
      }
      return null;
   }
}