Java, замена для бесконечных циклов?

Я делаю программу, которая показывает рост клеток через массив. Я получил это, когда я нажимаю кнопку запуска, массив обновляется через каждые 10 секунд (true) {} loop. Проблема в том, что я хочу, чтобы остановить цикл, нажав кнопку паузы, но, находясь в цикле, он не позволит мне использовать какие-либо элементы управления. Мне нужно что-то другое, кроме бесконечного цикла в orer, чтобы обновить фреймы.

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

Ответ 1

Вам нужно использовать Timer, который изменяет состояние вашего компонента (в данном случае рост сотовой связи), а затем вызывать JComponent.repaint()

Этот таймер можно отменить, чтобы сделать паузу, а затем перезапустить его, просто создайте новый:

Таким образом, вы можете определить следующие два метода:

private Timer timer;
...
public void startPaiting() {
    timer = new Timer();
    timer.schedule( new TimerTask(){
        public void run(){
            changeState();
            repaint();
        }
    },0,  10000 ); // 10 s. 
}

public void pause(){
    timer.cancel();
}

И затем в вашей кнопке "Пауза/Возобновить" вызовите следующие методы "пауза/начало":

if( e.getActionCommand().equals("Pause")){
    growPanel.pause();
    setText("Resume");
} else {
    growPanel.startPaiting();
    setText("Pause");
}

Вот полный исходный код, чтобы увидеть, как он работает:

import javax.swing.*;
import java.awt.Graphics;
import java.awt.Point;
import java.awt.event.*;
import java.util.Timer;
import java.util.TimerTask;

public class Grow {

    public static void main( String [] args ) {
        JFrame frame = new JFrame();
        final GrowPanel growPanel = new GrowPanel();
        frame.add( growPanel );
        frame.add( new JPanel(){{
            add( new JButton("Pause"){{
                addActionListener( new ActionListener(){
                    public void actionPerformed( ActionEvent e ){
                        if( e.getActionCommand().equals("Pause")){
                            growPanel.pause();
                            setText("Resume");
                        } else {
                            growPanel.startPaiting();
                            setText("Pause");
                        }
                    }
                });
        }});}}, java.awt.BorderLayout.SOUTH );
        frame.setSize( 400, 300 );
        frame.setVisible( true );
    }
}

class GrowPanel extends JComponent {
    private int x;
    private int y;
    private Timer timer;
    GrowPanel() {
        x = 10;
        y = 10;
        startPaiting();
    }

    public void startPaiting() {
        timer = new Timer();
        timer.schedule( new TimerTask(){
            public void run(){
                changeState();
                repaint();
            }
        },0,  100 ); // or 10000 which is 10 s. 
    }

    public void pause(){
        timer.cancel();
    }

    public void paintComponent( Graphics g ){
        g.fillOval( x, y, 10, 10 );
    }
    private void changeState(){
            x+=10;
            if( x >= 400 ) {
                y+=10;
                x = 0;
            }
            if( y >= 300 ){
                y = 10;
            }
    }

}

Ответ 2

Я бы предложил использовать отдельный поток для обработки массива. Убедитесь, что вы используете потокобезопасный объект (проверьте документы Java) и просто вызывайте .start() в свой объект потока, когда хотите начать. Держите указатель на него, чтобы вы могли приостановить его с помощью setPaused (true)

Что-то вроде этого....

class MyArrayUpdater extends Thread {
    private volatile boolean enabled = true;
    private volatile boolean paused = true;

    @Override
    public void run() {
        super.run();
        try {
            while (enabled) {
                if (!paused) {
                    // Do stuff to your array here.....
                }
                Thread.sleep(1);
            }
        } catch (InterruptedException ex) {
            // handle ex
        }
    }

    public void setEnabled(boolean arg) {
        enabled = arg;
    }

    public void setPaused(boolean arg) {
        paused = arg;
    }
}

Ответ 3

Если эти кнопки являются кнопками Swing, тогда способ сделать это: нажмите кнопку "Пуск", создайте новый объект javax.swing.Timer, который обновляет каждые 10 секунд. Затем нажмите кнопку "Пауза", чтобы остановить этот таймер.

Ответ 4

Вы хотите запустить симуляцию в потоке (Look for Runnable Interface). Затем вы можете передавать сообщения в эту тему для приостановки, продолжения и остановки.

Ответ 5

Я лично предпочитаю использовать класс Timer, а не Thread или Thread.sleep(). Класс таймера обрабатывает и запуск кода периодически, и его отмену.

Ваш код будет выглядеть так:

TimerTask myTask = new TimerTask() {
  public void run() {
    // here goes my code
  }
};

Timer timer = new Timer();
timer.schedule(myTask, 0 /*starts now*/, 10 * 1000 /* every 10 seconds */);

// whenever you need to cancel it:
timer.cancel();

Ответ 6

Угу. Похоже, вы делаете свой рисунок на потоке диспетчера событий. События (нажатие на кнопки) никогда не имеют возможности получить вызов, потому что вы никогда не возвращали управление с момента последнего вызова...

Ответ 7

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

Конечно, это также может привести к использованию семафора для связи между двумя потоками. На каком уровне вы участвуете? Вы уже затронули эти темы?

Ответ 8

Посмотрите на механизм wait()/notify(). Вкратце, поток может ждать 10 секунд, но все равно будет уведомлен о том, что произошло внешнее событие,

Ответ 9

Другой способ решения этой проблемы - использование BlockingQueue заданий.