Подтвердить что ты не робот

Остановить/отменить поток SwingWorker?

Я пытаюсь выяснить, как остановить поток SwingWorker от запуска, когда я нажимаю кнопку. Я смотрю вокруг, и у меня возникают проблемы с тем, как это сделать. На данный момент это то, что у меня есть:

new MySwingWorkerClass(args).execute();

Затем я создаю кнопку, которую я хочу использовать, чтобы остановить поток:

button = new JButton("Stop");
button.addActionListener(new ActionListener() {
        public void actionPerformed(ActionEvent e) 
        {
        // Stop the swing worker thread
        }
});

Я уже огляделся в поисках ответа, до сих пор мне удалось найти метод отмены. Я не понимаю, как использовать это, чтобы остановить моего качающегося работника. Я пробовал следующее, но это не сработало:

SwingWorker.cancel(true);
4b9b3361

Ответ 1

Вам нужно сохранить ссылку на ваш SwingWorker, тогда вы используете эту ссылку для отмены рабочего потока.

MySwingWorker myWorker = new MySwingWorkerClass(args).execute();

button.addActionListener(new ActionListener() {
    public void actionPerformed(ActionEvent e) 
    {
        // Stop the swing worker thread
        myWorker.cancel(true);
    }
});

Вот полный пример:

enter image description here

public class WorkerDemo extends JFrame {
    private boolean isStarted = false;
    private JLabel counterLabel = new JLabel("Not started");
    private Worker worker = new Worker();
    private JButton startButton = new JButton(new AbstractAction("Start") {

        @Override
        public void actionPerformed(ActionEvent arg0) {
            if(!isStarted) { 
                worker.execute();
                isStarted = false;
            }
        }

    });
    private JButton stopButton = new JButton(new AbstractAction("Stop") {

        @Override
        public void actionPerformed(ActionEvent arg0) {
            worker.cancel(true);
        }

    });

    public WorkerDemo() {

        add(startButton, BorderLayout.WEST);
        add(counterLabel, BorderLayout.CENTER);
        add(stopButton, BorderLayout.EAST);
        pack();
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);
    }

    class Worker extends SwingWorker<Void, Integer> {

        int counter = 0;

        @Override
        protected Void doInBackground() throws Exception {
            while(true) {
                counter++;
                publish(counter);
                Thread.sleep(60);
            }
        }

        @Override
        protected void process(List<Integer> chunk) {

            // get last result
            Integer counterChunk = chunk.get(chunk.size()-1);

            counterLabel.setText(counterChunk.toString());
        }

    }

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

            @Override
            public void run() {
                new WorkerDemo();
            }

        });
    }

}

Ответ 2

Вам необходимо периодически проверять его отмененный флаг (т.е. isCancelled()). SwingWorker выходит, как обрабатывать прерывание до вас.

Для получения дополнительной информации см. Отмена фоновых задач.

Ответ 3

вы можете попробовать переопределить метод done() и поместить его в try catch и поймать java.util.concurrent.CancellationException. Что-то вроде

. , .

@Overide
done() {
   try {
     get();
   } catch (CancellationException e) {
       // Do your task after cancellation
   }
}

. , .

Ответ 4

Я думаю, вам нужно перезапустить счетчик после каждой остановки.

Для startButton вы должны перезапустить рабочий, как в

private JButton startButton = new JButton(new AbstractAction("Start") {

        @Override
        public void actionPerformed(ActionEvent arg0) {
            if(!isStarted) { 
                worker = new Worker();     // New line
                worker.execute();
                isStarted = false;
            }
        }
    });

Чтобы остановить, вы можете использовать

worker.setStopFlag(); // in stopButton actionPerformed block.

в рабочем классе   private boolean stopFlag = false;

и добавьте

if( stopFlag)
break;

после Thread.sleep(60); и, наконец, поставить setter для stopFlag в конце класса Worker как

void setStopFlag(){
    stopFlag = true;
}

Кстати, вы можете использовать cancel (true), если хотите выйти из исключения.