gpt4 book ai didi

java - 递归需要不必要的返回并重新启动应用程序

转载 作者:太空宇宙 更新时间:2023-11-04 12:01:27 25 4
gpt4 key购买 nike

我用 Java 重新创建了游戏 2048,它确实有效(我没想到自己真的能做出这样“高级”的东西。)

当游戏中的瓷砖移动时,必须创建一个新的方 block /瓷砖/数字,因此代码通过随机选择 0 到 15 之间的数字来搜索可用位置(瓷砖已编号为 0 到 15(实际上它是一个具有 4 列和 4 行的多维数组))。选择一个随机点后,它会检查那里是否已经有一个正方形/方 block /数字,如果是,它会通过返回自身来调用此函数。因此,我现在也必须在没有被占用的情况下退回一些东西,所以我只是退回了一个我以后不用的号码。我做错了什么?

另外,为了开始一个新游戏,我现在基本上重置了 newGame(); 中的所有变量;功能。有没有办法重新启动应用程序或类似的东西?在网站上找不到任何内容。

我还想知道我是否遵守编程惯例,以及对我重新创建此练习的方法的一般评论。 (我知道这是否会占用您很多时间)。

代码:

import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;

public class Test2048 extends Application{

int score = 0;
Label scoreLabel = new Label("Score: 0");
Label statusLabel = new Label("");
int[][] cell;
Label[][] labels = new Label[4][4];

public static void main(String[] args) {launch(args);}

private void init(Stage primaryStage) {

primaryStage.setTitle("2048");

GridPane grid = new GridPane();
grid.setVgap(-10);
grid.setHgap(-10);
grid.setStyle("-fx-background-color: #BBADA0; -fx-background-radius: 30; -fx-max-width: 1; -fx-border-radius: 16; -fx-border-width: 10; -fx-border-color: #F4F4F4; ");
addLabels(grid);

newGame();
Button newGame = new Button("New Game");
newGame.setOnAction(e -> {
score = 0;
statusLabel.setText("");
newGame();
});

VBox layout = new VBox();
layout.getChildren().addAll(grid, scoreLabel, statusLabel, newGame);

Scene firstScene = new Scene(layout, 390, 450);
firstScene.setOnKeyPressed(e -> {
String arrow = e.getCode().toString();
switch(arrow){
case "RIGHT": move(0, 1, 2, 3, 4, 4, 4, 4); break;
case "DOWN": move(4, 4, 4, 4, 0, 1, 2, 3); break;
case "LEFT": move(3, 2, 1, 0, 4, 4, 4, 4); break;
case "UP": move(4, 4, 4, 4, 3, 2, 1, 0); break;
}
});
primaryStage.setScene(firstScene);

}

@Override
public void start(Stage primaryStage) throws Exception {
init(primaryStage);
primaryStage.show();
}

private void addLabels(GridPane grid) {
for (int x=0;x<4;x++){
for (int y=0;y<4;y++){
labels[x][y]=new Label();
grid.add(labels[x][y], x, y);
}
}
}

private void newGame(){
cell = new int[4][4];
newCell();
newCell();
updateScreen();
}

private void move(int ax, int bx, int cx, int dx, int ay, int by, int cy, int dy) {
boolean hor = false;
if (ax == 4){
hor = true;
}

Boolean hasMoved = false;
for (int i = 0; i <= 3; i++) {
int hasChanged;
if(hor){
ax = bx = cx = dx = i;
}else{
ay = by = cy = dy = i;
}

hasChanged = 4;
if (cell[cx][cy] != 0) {
if (cell[dx][dy] != 0) {
if (cell[cx][cy] == cell[dx][dy]) {
cell[cx][cy] = 0;
cell[dx][dy] *= 2;
score += cell[dx][dy];
hasChanged = 1;
hasMoved = true;
}
} else {
cell[dx][dy] = cell[cx][cy];
cell[cx][cy] = 0;
hasMoved = true;
}
}

if (cell[bx][by] != 0) {
if (cell[cx][cy] != 0) {
if (cell[bx][by] == cell[cx][cy]) {
cell[bx][by] = 0;
cell[cx][cy] *= 2;
score += cell[cx][cy];
hasChanged = 0;
hasMoved = true;
}
} else {
if (cell[dx][dy] != 0) {
if (cell[bx][by] == cell[dx][dy] && hasChanged != 1) {
cell[bx][by] = 0;
cell[dx][dy] *= 2;
score += cell[dx][dy];
hasChanged = 1;
hasMoved = true;
} else {
cell[cx][cy] = cell[bx][by];
cell[bx][by] = 0;
hasMoved = true;
}
} else {
cell[dx][dy] = cell[bx][by];
cell[bx][by] = 0;
hasMoved = true;
}
}
}

if (cell[ax][ay] != 0) {
if (cell[bx][by] != 0) {
if (cell[ax][ay] == cell[bx][by]) {
cell[ax][ay] = 0;
cell[bx][by] *= 2;
score += cell[bx][by];
hasMoved = true;
}
} else {
if (cell[cx][cy] != 0) {
if (cell[ax][ay] == cell[cx][cy] && hasChanged != 0) {
cell[ax][ay] = 0;
cell[cx][cy] *= 2;
score += cell[cx][cy];
hasMoved = true;
} else {
cell[bx][by] = cell[ax][ay];
cell[ax][ay] = 0;
hasMoved = true;
}
} else {
if (cell[dx][dy] != 0) {
if (cell[ax][ay] == cell[dx][dy] && hasChanged != 1) {
cell[ax][ay] = 0;
cell[dx][dy] *= 2;
score += cell[dx][dy];
hasMoved = true;
} else {
cell[cx][cy] = cell[ax][ay];
cell[ax][ay] = 0;
hasMoved = true;
}
} else {
cell[dx][dy] = cell[ax][ay];
cell[ax][ay] = 0;
hasMoved = true;
}
}
}
}
}

if(hasMoved) newCell();

checkStatus();
updateScreen();
}

private void checkStatus(){

long noSpace = 1;
boolean won = false;

for(int x=0;x<4;x++){
for(int y=0;y<4;y++){
noSpace *= cell[x][y];
if(cell[x][y] == 2048) won = true;
}
}
boolean alive = false;

if(noSpace != 0){
for(int x=0;x<3;x++) {
for(int y=0;y<4;y++) {
if (cell[x][y] == cell[x + 1][y]) {
alive = true;
}
}
}
for(int x=0;x<4;x++){
for(int y=0;y<3;y++){
if (cell[x][y] == cell[x][y+1]){
alive = true;
}
}
}
}else{
alive = true;
}

if (!alive){
statusLabel.setText("Game Over!");
} else if(won) {
statusLabel.setText("Won!");
}
}

private int newCell(){
int x = (int) (4 * Math.random());
int y = (int) (4 * Math.random());
if(cell[x][y] != 0){
return newCell();
} else {
cell[x][y] = twoOrFour();
return x; //I don't need this x anywhere, but I need to return newCell()
}
}

private int twoOrFour(){
double r = Math.random();
if(r < 0.125){
return 4;
}else{
return 2;
}
}

private void updateScreen() {

for(int x=0;x<4;x++){
for(int y=0;y<4;y++){
labels[x][y].setText(Integer.toString(cell[x][y]));
labels[x][y].setStyle(cellStyle(cell[x][y]));
}
}

scoreLabel.setText("Score: " + Integer.toString(score));
}

private String cellStyle(int cell) {
String style = "-fx-min-width: 100; -fx-background-radius: 20; -fx-font: bolder 30 'ClearSans'; -fx-min-height: 100; -fx-alignment: center; -fx-border-radius: 14; -fx-border-width: 10; -fx-border-color: #BBADA0; -fx-background-color: #";
switch(cell){
case 2: style+= "EEE4DA"; break;
case 4: style+= "ECE0C8"; break;
case 8: style+= "F2B179"; break;
case 16: style+= "F59563"; break;
case 32: style+= "F57C5F"; break;
case 64: style+= "F65D3B"; break;
case 128: style+= "EDCE71"; break;
case 256: style+= "EDCC61"; break;
case 512: style+= "ECC850"; break;
case 1024: style+= "EDC53F"; break;
case 2048: style+= "ECC400"; break;
case 0: style+= "CDC1B4"; break;
default: style+= "000000"; break;
}

style+= ";-fx-text-fill: #";

if(cell==2||cell==4){
style+= "776E65;";
}else if(cell==0){
style+= "CDC1B4;";
}else{
style+= "F9F6F2;";
}
return style;
}
}

最佳答案

正如@RyanJ 的评论,您只需使用 void 返回类型定义 newCell():

private void newCell() {
int x = (int)(Math.random() * 4);
int y = (int)(Math.random() * 4);
if (cell[x][y] != 0) {
newCell();
} else {
cell[x][y] = twoOrFour();
}
}

注意你可以用一个循环做同样的事情:

private void newCell() {
int x ;
int y ;
do {
x = (int)(Math.random() * 4);
y = (int)(Math.random() * 4);
} while (cell[x][y] != 0);
cell[x][y] = twoOrFour();
}

不过,这两种方法都不是特别好,因为找到空白空间可能需要任意长的时间。在递归版本中,情况更糟,因为您可能运气不好,递归运行得足够深,导致 StackOverflowException。 (这不太可能发生,但如果你玩的时间足够长(让我们面对现实吧,这个游戏会让人上瘾),它总有一天会发生。)

这是一个更好的解决方案。创建一个空单元格列表(只是一个介于 0 和 15 之间的整数列表)。然后随机选择一个:

private void newCell() {
List<Integer> availableCells = new ArrayList<>();
for (int i = 0; i < 16; i++) {
int x = i / 4 ;
int y = i % 4 ;
if (cell[x][y] == 0) {
availableCells.add(i);
}
}
int nextCell = availableCells.get((int)(Math.random() * availableCells.size()));
cell[nextCell / 4][nextCell % 4] = twoOrFour();
}

关于java - 递归需要不必要的返回并重新启动应用程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29759446/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com