Java SwingWorker not terminating on task completion

前端 未结 2 1707
陌清茗
陌清茗 2021-01-23 09:07

Ok, so I\'ve been playing around with SwingWorker a bit and got some simplified code for updating a gui going, but I\'m having trouble figuring out how to get the thread to prop

相关标签:
2条回答
  • 2021-01-23 09:38

    for example

    import javax.swing.*;
    import java.awt.*;
    import java.awt.event.*;
    import java.util.ArrayList;
    
    public class SwingWorkerExample extends JFrame implements ActionListener {
    
        private static final long serialVersionUID = 1L;
        private final JButton startButton, stopButton;
        private JScrollPane scrollPane = new JScrollPane();
        private JList listBox = null;
        private DefaultListModel listModel = new DefaultListModel();
        private final JProgressBar progressBar;
        private mySwingWorker swingWorker;
    
        public SwingWorkerExample() {
            super("SwingWorkerExample");
            setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            getContentPane().setLayout(new GridLayout(2, 2));
            startButton = makeButton("Start");
            stopButton = makeButton("Stop");
            stopButton.setEnabled(false);
            progressBar = makeProgressBar(0, 99);
            listBox = new JList(listModel);
            scrollPane.setViewportView(listBox);
            getContentPane().add(scrollPane);
            //Display the window.
            pack();
            setVisible(true);
        }
    //Class SwingWorker<T,V> T - the result type returned by this SwingWorker's doInBackground
    //and get methods V - the type used for carrying out intermediate results by this SwingWorker's 
    //publish and process methods
    
        private class mySwingWorker extends javax.swing.SwingWorker<ArrayList<Integer>, Integer> {
    //The first template argument, in this case, ArrayList<Integer>, is what s returned by doInBackground(), 
    //and by get(). The second template argument, in this case, Integer, is what is published with the 
    //publish method. It is also the data type which is stored by the java.util.List that is the parameter
    //for the process method, which recieves the information published by the publish method.
    
            @Override
            protected ArrayList<Integer> doInBackground() {
    //Returns items of the type given as the first template argument to the SwingWorker class.
                if (javax.swing.SwingUtilities.isEventDispatchThread()) {
                    System.out.println("javax.swing.SwingUtilities.isEventDispatchThread() returned true.");
                }
                Integer tmpValue = new Integer(1);
                ArrayList<Integer> list = new ArrayList<Integer>();
                for (int i = 0; i < 100; i++) {
                    for (int j = 0; j < 100; j++) { //find every 100th prime, just to make it slower
                        tmpValue = FindNextPrime(tmpValue.intValue());
    //isCancelled() returns true if the cancel() method is invoked on this class. That is the proper way
    //to stop this thread. See the actionPerformed method.
                        if (isCancelled()) {
                            System.out.println("SwingWorker - isCancelled");
                            return list;
                        }
                    }
    //Successive calls to publish are coalesced into a java.util.List, which is what is received by process, 
    //which in this case, isused to update the JProgressBar. Thus, the values passed to publish range from 
    //1 to 100.
                    publish(new Integer(i));
                    list.add(tmpValue);
                }
                return list;
            }//Note, always use java.util.List here, or it will use the wrong list.
    
            @Override
            protected void process(java.util.List<Integer> progressList) {
    //This method is processing a java.util.List of items given as successive arguments to the publish method.
    //Note that these calls are coalesced into a java.util.List. This list holds items of the type given as the
    //second template parameter type to SwingWorker. Note that the get method below has nothing to do with the 
    //SwingWorker get method; it is the List's get method. This would be a good place to update a progress bar.
                if (!javax.swing.SwingUtilities.isEventDispatchThread()) {
                    System.out.println("javax.swing.SwingUtilities.isEventDispatchThread() + returned false.");
                }
                Integer percentComplete = progressList.get(progressList.size() - 1);
                progressBar.setValue(percentComplete.intValue());
            }
    
            @Override
            protected void done() {
                System.out.println("doInBackground is complete");
                if (!javax.swing.SwingUtilities.isEventDispatchThread()) {
                    System.out.println("javax.swing.SwingUtilities.isEventDispatchThread() + returned false.");
                }
                try {
    //Here, the SwingWorker's get method returns an item of the same type as specified as the first type parameter
    //given to the SwingWorker class.
                    ArrayList<Integer> results = get();
                    for (Integer i : results) {
                        listModel.addElement(i.toString());
                    }
                } catch (Exception e) {
                    System.out.println("Caught an exception: " + e);
                }
                startButton();
            }
    
            boolean IsPrime(int num) { //Checks whether a number is prime
                int i;
                for (i = 2; i <= num / 2; i++) {
                    if (num % i == 0) {
                        return false;
                    }
                }
                return true;
            }
    
            protected Integer FindNextPrime(int num) { //Returns next prime number from passed arg.       
                do {
                    if (num % 2 == 0) {
                        num++;
                    } else {
                        num += 2;
                    }
                } while (!IsPrime(num));
                return new Integer(num);
            }
        }
    
        private JButton makeButton(String caption) {
            JButton b = new JButton(caption);
            b.setActionCommand(caption);
            b.addActionListener(this);
            getContentPane().add(b);
            return b;
        }
    
        private JProgressBar makeProgressBar(int min, int max) {
            JProgressBar progressBar1 = new JProgressBar();
            progressBar1.setMinimum(min);
            progressBar1.setMaximum(max);
            progressBar1.setStringPainted(true);
            progressBar1.setBorderPainted(true);
            getContentPane().add(progressBar1);
            return progressBar1;
        }
    
        private void startButton() {
            startButton.setEnabled(true);
            stopButton.setEnabled(false);
            System.out.println("SwingWorker - Done");
        }
    
        @Override
        public void actionPerformed(ActionEvent e) {
            if ("Start" == null ? e.getActionCommand() == null : "Start".equals(e.getActionCommand())) {
                startButton.setEnabled(false);
                stopButton.setEnabled(true);
    // Note that it creates a new instance of the SwingWorker-derived class. Never reuse an old one.
                (swingWorker = new mySwingWorker()).execute(); // new instance
            } else if ("Stop" == null ? e.getActionCommand() == null : "Stop".equals(e.getActionCommand())) {
                startButton.setEnabled(true);
                stopButton.setEnabled(false);
                swingWorker.cancel(true); // causes isCancelled to return true in doInBackground
                swingWorker = null;
            }
        }
    
        public static void main(String[] args) {
    // Notice that it kicks it off on the event-dispatching thread, not the main thread.
            SwingUtilities.invokeLater(new Runnable() {
    
                @Override
                public void run() {
                    SwingWorkerExample swingWorkerExample = new SwingWorkerExample();
                }
            });
        }
    }
    
    0 讨论(0)
  • 2021-01-23 09:47

    I'm not sure what you're trying to achieve. Your example is working fine. Worker thread runs till the end. In case you want to wait till it ends to do something, you have to call method pbTask.get() somewhere in your code. Otherwise, it will just quietly completes without affecting any of your UI components.

    Consider the following change to your method to see how it behaves now. Note, that UI freezes because you make UI wait for the thread to complete, but output "DONE" appears in your log only when the WorkerThread completes.

    public void actionPerformed(ActionEvent e) {
        if ("Start" == e.getActionCommand() && pbTask == null) {
            theButton.setText("Stop");
            theButton.setActionCommand("Stop");
            (pbTask = new PBTask()).execute();
        } else if ("Stop" == e.getActionCommand()) {
            theButton.setText("Start");
            theButton.setActionCommand("Start");
            pbTask.cancel(true);
            pbTask = null;
        } else {
            alertMsg("Thread still running.");
        }
        try {
            pbTask.get();
        } catch (InterruptedException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        } catch (ExecutionException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        }
        System.out.println("DONE");
    }
    

    This change is just to illustrate the difference. In order to write actual code we need to know more about what you're trying to achieve.


    If my extrasensory skills are ok, then you probably want to flip button back to "Start". In order to do this, you need to override done() method in the Worker:

    private class PBTask extends SwingWorker<Void, UpdatePB> {
        @Override
        protected Void doInBackground() {
            int prog1 = 0;
            int prog2 = 0;
            Random random = new Random();
    
            while (prog2 < 100) {
                if(prog1 >= 100) {
                    prog1 = 0;
                }
                //Sleep for up to one second.
                try {
                    Thread.sleep(random.nextInt(100));
                } catch (InterruptedException ignore) {}
                //Make random progress.
                prog1 += random.nextInt(10);
                prog2 += random.nextInt(5);
                publish(new UpdatePB(prog1, prog2));
            }
            return null;
        }
    
        @Override
        protected void process(List<UpdatePB> pairs) {
            UpdatePB pair = pairs.get(pairs.size() - 1);
                pb.setValue(pair.pb1);
                pbF.setValue(pair.pb2);
        }
    
        @Override
        protected void done() {
            super.done();
            theButton.setText("Start");
            theButton.setActionCommand("Start");
        }
    }
    
    0 讨论(0)
提交回复
热议问题