java - 当 Swing Timer 启动时,ActionListener 类不断重复

标签 java swing timer

我正在尝试使用 Java GUI 开发一个非常基本的“Simon said”模拟器。我有一个生成并返回 int[] 数组的方法;对于数组中的每个元素,计时器 computer 应启动,为指定的 JButton 调用 doClick() 方法,并等待 1/2 秒。每个 JButton 都连接到一个 ActionListener(),它将特定按钮的颜色更改为白色,激活另一个计时器 timer,并更改按钮回到原来的颜色。

每次我在 for 循环中调用 computer.start(); 时,它都会运行 ComputerListener() 中的代码,但它会无限重复。我添加了打印语句,以便我可以通过 Netbeans 上的输出看到发生了什么。我在论坛上看过类似的问题,但没有提供可行的解决方案。

我的问题:为什么在 for 循环中调用 computer.start(); 时,我的 ComputerListener 类会重复?

package simon;

// @jagged_prospect

import java.util.Random;
import java.util.Arrays;
import java.awt.event.*;
import java.awt.*;
import javax.swing.*;
import javax.swing.plaf.ButtonUI;
import javax.swing.plaf.basic.BasicButtonUI;

public class SIMONPanel extends JPanel{

private static final int PANEL_W=300,PANEL_H=300;
private static final int PREF_W=500,PREF_H=500;
private static final String[] CARD_LABELS={"main","info","game"};

private final JPanel gameCard,infoCard,splashCard;
private final JButton rButton,yButton,gButton,bButton;
private final int lives=3;

private CardLayout cardlayout=new CardLayout();
private JPanel cards=new JPanel(cardlayout);
private Action[] actions={new ShowMainAction(),new ShowInfoAction(),
    new ShowGameAction()};
private Object source;
private Timer timer,computer;

public SIMONPanel(){
    setBackground(Color.BLACK);
    setLayout(new BorderLayout());
    gameCard=new JPanel();
    infoCard=new JPanel();
    splashCard=new JPanel();

    // game card panel
    gameCard.setLayout(new BorderLayout());
    gameCard.setPreferredSize(new Dimension(PANEL_W,PANEL_H));

    JPanel gameButtonPanel=new JPanel();
    gameButtonPanel.setLayout(new GridLayout(2,2));
    JButton startButton=new JButton("Start");
    startButton.addActionListener(new StartListener());

    rButton=new JButton("red");
    rButton.addActionListener(new ColorButtonListener());
    rButton.setSize(50,50);
    rButton.setUI((ButtonUI)BasicButtonUI.createUI(rButton));
    rButton.setBackground(Color.RED);
    rButton.setForeground(Color.WHITE);

    yButton=new JButton("yellow");
    yButton.addActionListener(new ColorButtonListener());
    yButton.setSize(50,50);
    yButton.setUI((ButtonUI)BasicButtonUI.createUI(yButton));
    yButton.setBackground(Color.YELLOW);

    gButton=new JButton("green");
    gButton.addActionListener(new ColorButtonListener());
    gButton.setSize(50,50);
    gButton.setUI((ButtonUI)BasicButtonUI.createUI(gButton));
    gButton.setBackground(Color.GREEN);

    bButton=new JButton("blue");
    bButton.addActionListener(new ColorButtonListener());
    bButton.setSize(50,50);
    bButton.setUI((ButtonUI)BasicButtonUI.createUI(bButton));
    bButton.setBackground(Color.BLUE);
    bButton.setForeground(Color.WHITE);

    gameButtonPanel.add(gButton);
    gameButtonPanel.add(rButton);
    gameButtonPanel.add(yButton);
    gameButtonPanel.add(bButton); 
    gameCard.add(gameButtonPanel,BorderLayout.CENTER);
    gameCard.add(startButton,BorderLayout.SOUTH);

    // splash card panel
    splashCard.setLayout(new BorderLayout());
    splashCard.setPreferredSize(new Dimension(PANEL_W,PANEL_H));
    splashCard.setBackground(Color.BLACK);
    JLabel titleLabel=new JLabel("S I M O N",SwingConstants.CENTER);
    titleLabel.setFont(new Font("Niagara Solid",Font.BOLD,84));
    titleLabel.setForeground(Color.WHITE);
    splashCard.add(titleLabel,BorderLayout.CENTER);

    // info card panel
    // nothing here yet

    JPanel buttonPanel=new JPanel(new GridLayout(1,0,5,0));
    for(Action action : actions){
        buttonPanel.add(new JButton(action));
        buttonPanel.setBackground(Color.BLACK);
    }

    cards.add(splashCard,CARD_LABELS[0]);
    cards.add(infoCard,CARD_LABELS[1]);
    cards.add(gameCard,CARD_LABELS[2]);

    add(cards,BorderLayout.CENTER);
    add(buttonPanel,BorderLayout.SOUTH);
}
// sets uniform panel size
    @Override
public Dimension getPreferredSize() {
    return new Dimension(PREF_W, PREF_H);
}
// shows the Main Menu card
private class ShowMainAction extends AbstractAction {
    public ShowMainAction() {
        super("Main");
    }
    @Override
    public void actionPerformed(ActionEvent e) {
        cardlayout.show(cards,CARD_LABELS[0]);
    } 
}
// shows the Info card
private class ShowInfoAction extends AbstractAction {
    public ShowInfoAction() {
        super("Info");
    }
    @Override
    public void actionPerformed(ActionEvent e) {
        cardlayout.show(cards,CARD_LABELS[1]);
    }
}
// show the Game card
private class ShowGameAction extends AbstractAction {
    public ShowGameAction() {
        super("Game");
    }
    @Override
    public void actionPerformed(ActionEvent e) {
        cardlayout.show(cards,CARD_LABELS[2]);
    }
}
private class TimerListener implements ActionListener{
    @Override
    public void actionPerformed(ActionEvent event){
        if(source==gButton){
            gButton.setBackground(Color.GREEN);
        }
        else if(source==rButton){
            rButton.setBackground(Color.RED);
            rButton.setForeground(Color.WHITE);
        }
        else if(source==yButton){
            yButton.setBackground(Color.YELLOW);
        }
        else if(source==bButton){
            bButton.setBackground(Color.BLUE);
            bButton.setForeground(Color.WHITE);
        } 
    }
}
private class ColorButtonListener implements ActionListener{
    @Override
    public void actionPerformed(ActionEvent event){
        source=event.getSource();
        int delay=300;
        timer=new Timer(delay,new TimerListener());

        if(source==gButton){
            gButton.setBackground(Color.WHITE);
            timer.setRepeats(false);
            timer.start();
        }
        else if(source==rButton){
            rButton.setBackground(Color.WHITE);
            rButton.setForeground(Color.BLACK);
            timer.setRepeats(false);
            timer.start();
        }
        else if(source==yButton){
            yButton.setBackground(Color.WHITE);
            timer.setRepeats(false);
            timer.start();
        }
        else if(source==bButton){
            bButton.setBackground(Color.WHITE);
            bButton.setForeground(Color.BLACK);
            timer.setRepeats(false);
            timer.start();
        }     
    }
}
private class StartListener implements ActionListener{
    public void actionPerformed(ActionEvent event){
        // calls generateSequence() to make pattern for player to replicate
        // for debugging in output
        System.out.println(Arrays.toString(generateSequence()));
    }
}
public int[] generateSequence(){
    Random ran=new Random();
    ComputerListener cpu=new ComputerListener();
    computer=new javax.swing.Timer(500,cpu);
    int seqLen=4;
    int[] gameSequence=new int[seqLen];
    for(int x=0;x<seqLen;x++){
       int assign=ran.nextInt(4)+1;
       gameSequence[x]=assign;
    }
    for(int y=0;y<seqLen;y++){ // print and wait 1/2 second, repeat 3 times
        computer.start();
    }
    //computer.stop(); // should stop ComputerListener()???
    return gameSequence;
}
private class ComputerListener implements ActionListener{
    public void actionPerformed(ActionEvent event){
        // for debugging in output
        System.out.println("it worked");
    }
}
}

最佳答案

您在 for 循环中多次调用计算机 Swing Timer 的开始按钮,这不是您想要做的,事实上,计时器的全部目的是帮助你摆脱了for循环。相反,计时器会重复一个操作,并更改状态,并继续执行直到完成。考虑使用 int 数组或更好的 ArrayList 来保存计时器应迭代的颜色,并在该 ActionListener 中执行操作并将指针前进到数组或列表中的下一个位置,使用该指针来决定要执行的操作接下来做。然后当指针完全穿过集合时,停止计时器。

有关我所描述的确切示例,请在此处查看我的计时器的 ActionListener 以获取不完整的 Simon 游戏:Method keeps window from closing

计时器的 ActionListener(带注释)如下:

private class TimerListener implements ActionListener {
    private SimonPanel simonPanel;  // the Simon JPanel
    private int colorListIndex = 0; // index into the ArrayList of MyColor objects
    private int sliceCount = 0;
    private List<MyColor> myColorList; // the MyColor ArrayList -- the random colors to press
    private int maxCount;

    public TimerListener(SimonPanel simonPanel, List<MyColor> myColorList) {
        // pass in the key fields into the program via constructor parameter
        this.simonPanel = simonPanel;
        this.myColorList = myColorList;  // again the ArrayList that holds random MyColor objects
        maxCount = myColorList.size();  // size of my list
    }

    @Override
    public void actionPerformed(ActionEvent evt) {
        // if index at the end of the list -- get out and clean up
        if (colorListIndex == maxCount) {
            // clear the display of "pressed" colors
            for (MyColor myColor : MyColor.values()) {
                simonPanel.setMyColorPressed(myColor, false);
            }

            // stop this timer
            ((Timer) evt.getSource()).stop();
            return;
        }

        // the listener is a little complex since it must turn on colors and turn them off 
        // which is why I use a sliceCount int counter variable here
        if (sliceCount == 0) {
            // turn on the next color in the list (using the index)
            MyColor myColor = myColorList.get(colorListIndex);
            simonPanel.setMyColorPressed(myColor, true);
            sliceCount++;
        } else if (sliceCount < TIME_SLICES - 1) {
            sliceCount++;
            return;
        } else if (sliceCount == TIME_SLICES - 1) {
            sliceCount = 0;
            MyColor myColor = myColorList.get(colorListIndex);
            simonPanel.setMyColorPressed(myColor, false);  // turn off the color 
            colorListIndex++;  // and increment the index
            return;
        }
    }
}

关于java - 当 Swing Timer 启动时,ActionListener 类不断重复,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51353899/

相关文章:

java - Android studio : CutomListView gives strange output, 它应该提供 firebase 数据。有什么解决办法吗?

java - 设置 Swing 程序的默认字体

c# - C#中的毫秒级精确定时器

java - 如何获取控制台字符集?

java - 如何执行并发安全的映射对象操作?

java - 如何在循环的服务器请求中使用 Phaser?

java - 将 JFrame 中的所有组件调整为窗口的实际大小

java - 我如何编写代码来显示每个 JPanel 中的每个字母,以及如何旋转。 (JFrame、NetBeans)

c# - 定时器可以自动收集垃圾吗?

multithreading - 让 Delphi TTimer 与多线程应用程序一起工作