无限循环并计算java中的循环错误

时间:2015-11-04 03:22:18

标签: java loops for-loop infinite-loop counting

我试图创造游戏nim。我做到了这样,计算机可以随机从随机行中取出1-3张卡片。是的,计算机不会这样做,但是当只剩下一到三张卡时,它将进入无限循环。

主文件。不需要触及

package nimgame;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
/**
 *
 * @author Angela
 */
public class NimApp extends JFrame implements ActionListener {

    private static final int ROWS = 3;
    private JTextField[] gameFields; // Where sticks for each row shown
    private JTextField rowField;     // Where player enters row to select
    private JTextField sticksField;  // Where player enters sticks to take
    private JButton playButton;      // Pressed to take sticks
    private JButton AIButton;        // Pressed to make AI's move

    private NimGame nim;

    public NimApp() {

        // Build the fields for the game play 
        rowField = new JTextField(5);
        sticksField = new JTextField(5);
        playButton = new JButton("PLAYER");
        AIButton = new JButton("COMPUTER");
        playButton.addActionListener(this);
        AIButton.addActionListener(this);
        AIButton.setEnabled(false);

        // Create the layout
        JPanel mainPanel = new JPanel(new BorderLayout());
        getContentPane().add(mainPanel);

        JPanel sticksPanel = new JPanel(new GridLayout(3, 1));
        mainPanel.add(sticksPanel, BorderLayout.EAST);

        JPanel playPanel = new JPanel(new GridLayout(3, 2));
        mainPanel.add(playPanel, BorderLayout.CENTER);

        // Add the fields to the play panel
        playPanel.add(new JLabel("Row: ", JLabel.RIGHT));
        playPanel.add(rowField);
        playPanel.add(new JLabel("Sticks: ", JLabel.RIGHT));
        playPanel.add(sticksField);
        playPanel.add(playButton);
        playPanel.add(AIButton);

        // Build the array of textfields to display the sticks
        gameFields = new JTextField[ROWS];
        for (int i = 0; i < ROWS; i++) {
            gameFields[i] = new JTextField(10);
            gameFields[i].setEditable(false);
            sticksPanel.add(gameFields[i]);
        }
        setSize(350, 150);
        setVisible(true);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        nim = new NimGame(new int[]{3, 5, 7});
        draw();
    }

    // Utility function to redraw game
    private void draw() {
        for (int row = 0; row < ROWS; row++) {
            String sticks = "";
            for (int j = 0; j < nim.getRow(row); j++) {
                sticks += "|   ";
            }
            gameFields[row].setText(sticks);
        }
        rowField.setText("");
        sticksField.setText("");
    }

    public void actionPerformed(ActionEvent e) {

        // Player move
        if (e.getSource() == playButton) {

            // Get the row and number of sticks to take
            int row = Integer.parseInt(rowField.getText())-1;
            int sticks = Integer.parseInt(sticksField.getText());

            // Play that move
            nim.play(row, sticks);

            // Redisplay the board and enable the AI button
            draw();
            playButton.setEnabled(false);
            AIButton.setEnabled(true);

            // Determine whether the game is over
            if (nim.isOver()) {
                JOptionPane.showMessageDialog(null, "You win!");
                playButton.setEnabled(false);
            }
        }

        // Computer move
        if (e.getSource() == AIButton) {

            // Determine computer move
            nim.AIMove();

            // Redraw board
            draw();
            AIButton.setEnabled(false);
            playButton.setEnabled(true);

            // Is the game over?
            if (nim.isOver()) {
                JOptionPane.showMessageDialog(null, "You win!");
                playButton.setEnabled(false);
            }
        }
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        NimApp a = new NimApp();
    }
}

这是主文件的支持类

package nimgame;
import java.util.Random;
/**
 *
 * @author Angela
 */
public class NimGame { 
     int[] Sticks; //creating an array of sticks
     int totalSticks=0;

    public NimGame(int[] initialSticks){
    Sticks = initialSticks;}

    public int getRow(int r){
        return Sticks[r];}

    public void play(int r, int s){
    Sticks[r]=Sticks[r]-s;}

这应该算上板上留下的所有木棒。如果剩下1个棒,它应该返回true,如果板上有多个棒,它应该返回false。我在使用for循环时遇到了麻烦,并且正确计算了它。

public boolean isOver(){
    int theTotal = 0;
    for (int i =1; i<Sticks.length; i++){
         theTotal = totalSticks + Sticks[i];
            System.out.println(totalSticks);}
       totalSticks = theTotal;
       System.out.println(totalSticks);
        if (totalSticks<=1){
            return true;
        }
        return false;

该方法应该随机生成一行和一定数量的棒以丢弃计算机。这是我无限循环的地方。

public void AIMove(){
    Random randomInt = new Random ();
    int RandomRow = randomInt.nextInt(3);
    int RandomDiscard = randomInt.nextInt(3-0)+1;
    int randomize=0;



    while (RandomDiscard >Sticks[RandomRow] && totalSticks>1){
        RandomRow = randomInt.nextInt(3);
        RandomDiscard = randomInt.nextInt(3-0)+1;
        Sticks[RandomRow]=Sticks[RandomRow]-RandomDiscard;
        randomize = 1;}

    if (randomize==0)
    Sticks[RandomRow]=Sticks[RandomRow]-RandomDiscard;

    if (totalSticks <= 1){
    Sticks[RandomRow]=Sticks[RandomRow]-1;}
    isOver();
    }

            }

2 个答案:

答案 0 :(得分:2)

终止while循环的条件是,当您在循环体上RandomDiscard > Sticks[RandomRow]totalSticks>1时,Sticks[RandomRow]的值变小至少一,保持这种状况。

要打破身体上的循环,您需要增加Sticks[RandomRow]或减少totalSticks或将条件更改为RandomDiscard < Sticks[RandomRow] || totalSticks>1

编辑(下午10时18分): 看看问题和解决方案,我得到了一个更好的解决问题的方法就是我所做的:

public void AIMove(){
    Random randomInt = new Random ();

    boolean tryRemove =  true;

    while(tryRemove && totalSticks > 1){
        int RandomRow = randomInt.nextInt(3);
        if(Sticks[RandomRow] <= 0)//the computer can't remove from this row
            continue;

        //the max number to remove from row
        int size = 3;
        if( Sticks[RandomRow] < 3)//this row have lest that 3 cards
            size = Sticks[RandomROw];//make the max number to remove from the row be the number of cards on the row 
        int RandomDiscard = randomInt.nextInt(size) + 1;
        Sticks[RandomRow] = Sticks[RandomRow] - RandomDiscard;
        //I don't know if this is needed, but since we remove a RandomDiscard amount lest decrease the totalSticks
        totalSticks = totalSticks - RandomDiscard;
        //exit loop
        tryRemove = false;
    }

    if(totalSticks <= 1){
        Sticks[RandomRow]=Sticks[RandomRow]-1;
        isOver();
    }


}

我所做的是将随机循环和if语句组合在一个循环中。

答案 1 :(得分:0)

theTotal = totalSticks + Sticks[i];

应该是

theTotal = theTotal + Sticks[i];

此外,for循环可能应该从索引0而不是1开始:

for (int i =1; i<Sticks.length; i++){