我正在用Java做一个国际象棋游戏,并且(我认为)已经成功地为AI玩家实现了Negamax。我在添加阿尔法贝塔剪枝来改进算法时遇到了一些麻烦。我已经尝试了下面的教程和示例代码,但就是不明白它是如何工作的。
以下是我目前必须获得最佳移动的代码:
private Move getBestMove() {
System.out.println("Getting best move");
System.out.println("Thinking...");
List<Move> validMoves = generateMoves(true);
int bestResult = Integer.MIN_VALUE;
Move bestMove = null;
for (Move move : validMoves) {
executeMove(move);
System.out.println("Evaluating: " + move);
int evaluationResult = -evaluateNegaMax(this.lookForward, "", Integer.MIN_VALUE, Integer.MAX_VALUE);
undoMove(move);
if (evaluationResult > bestResult) {
bestResult = evaluationResult;
bestMove = move;
}
}
System.out.println("Done thinking! The best move is: " + bestMove);
return bestMove;
}
这是我尝试将aplha-beta修剪添加到我的(工作)内切方法中:
public int evaluateNegaMax(int lookForward, String indent, int alpha, int beta) {
if (lookForward <= 0
|| this.chessGame.getGameState() == ChessGame.GAME_STATE_WHITE_WON
|| this.chessGame.getGameState() == ChessGame.GAME_STATE_BLACK_WON) {
return evaluateState();
}
List<Move> moves = generateMoves(false);
for (Move currentMove : moves) {
System.out.println(indent + "Handling move: " + currentMove + " : " + alpha);
if (currentMove == null) {
continue;
}
executeMove(currentMove);
alpha = Math.max(alpha, -evaluateNegaMax(lookForward-1, " ", -beta, -alpha));
if (alpha > beta) {
break;
}
undoMove(currentMove);
}
return alpha;
}
最后是控制台的外观
Starting game flow
Looking 2 moves aheadExecuted: E/2 -> E/4
Tested 0 moves
Getting best move
Thinking...
Evaluating: B/8 -> A/6
Handling move: B/1 -> A/3 : -2147483648
Handling move: A/8 -> B/8 : -2147483647
Handling move: B/1 -> C/3 : 2
Handling move: B/8 -> A/8 : -2147483647
Handling move: A/6 -> B/4 : -3
Handling move: A/6 -> C/5 : -3
Handling move: G/8 -> F/6 : -2
Handling move: D/1 -> E/2 : 2
Handling move: B/8 -> A/8 : -2147483647
Handling move: D/1 -> F/3 : 2
Handling move: A/8 -> B/8 : -2147483647
Handling move: A/8 -> B/8 : -2147483647
Handling move: F/6 -> E/4 : -32
Handling move: F/6 -> G/4 : -17
Handling move: F/6 -> D/5 : -17
Handling move: G/1 -> E/2 : 2
Handling move: B/1 -> A/3 : -2147483647
Handling move: B/1 -> C/3 : -29
Handling move: E/1 -> F/1 : -28
Handling move: E/2 -> G/1 : -19
Handling move: E/2 -> C/3 : -19
Handling move: E/2 -> G/3 : -19
Handling move: E/2 -> D/4 : -19
Handling move: G/1 -> F/3 : 19
Handling move: A/8 -> B/8 : -2147483647
Handling move: G/1 -> H/3 : 19
Handling move: B/8 -> B/2 : -2147483647
Exception in thread "Thread-2" java.lang.NullPointerException
at Chess.logic.ChessGame.movePiece(ChessGame.java:166)
at Chess.ai.AiPlayerHandler.executeMove(AiPlayerHandler.java:158)
at Chess.ai.AiPlayerHandler.evaluateNegaMax(AiPlayerHandler.java:84)
at Chess.ai.AiPlayerHandler.getBestMove(AiPlayerHandler.java:47)
at Chess.ai.AiPlayerHandler.getMove(AiPlayerHandler.java:31)
at Chess.logic.ChessGame.waitForMove(ChessGame.java:125)
at Chess.logic.ChessGame.startGame(ChessGame.java:95)
at Chess.logic.ChessGame.run(ChessGame.java:338)
at java.lang.Thread.run(Thread.java:745)
任何帮助都将不胜感激。提前感谢。
我想我让它工作了。如果有任何关注此问题的人正在等待回复,代码如下:
public int evaluateNegaMax(int depth, String indent, int alpha, int beta) {
if (depth <= 0
|| this.chessGame.getGameState() == ChessGame.GAME_STATE_WHITE_WON
|| this.chessGame.getGameState() == ChessGame.GAME_STATE_BLACK_WON) {
return evaluateState();
}
List<Move> moves = generateMoves(false);
int bestValue = Integer.MIN_VALUE;
for (Move currentMove : moves) {
executeMove(currentMove);
int value = -evaluateNegaMax(depth - 1, indent + " ", -beta, -alpha);
System.out.println(indent + "Handling move: " + currentMove + " : " + value);
undoMove(currentMove);
counter++;
if (value > bestValue) {
bestValue = value;
}
if (bestValue > alpha) {
alpha = bestValue;
}
if (bestValue >= beta) {
break;
}
}
System.out.println(indent + "max: " + alpha);
return alpha;
}
我有以下奥赛罗(reversi)游戏的阿尔法-贝塔极小值的实现。我已经修复了这个线程中的一些问题。这一次我想改进这个函数的性能。MAX_DEPTH=8需要很长的时间。在保持AI有点体面的同时,可以做些什么来加快性能? } 实用功能:
我使用Lombok项目为字符串字段生成getter/setter。此字段(例如密码)具有验证注释。 我想在setter中添加trim方法,以便不计算长度中的空白。 如何在Lombok setter中添加修剪方法?或者我必须编写自定义setter?
使用“管理字体”对话框在 Dreamweaver 中添加和修改字体组合。 字体组合确定浏览器显示网页中的文本的方式。浏览器使用用户系统上安装的字体组合中的第一种字体;如果未安装字体组合中的任何一种字体,则浏览器按用户的浏览器首选参数指定的方式显示文本。 向字体列表添加 Adobe Edge Web Fonts 可在网页中使用 Adobe Edge Web Fonts。在页面中使用 Edge 字体时
问题内容: 问题发生在 包含该行的代码位于 所有这一切都在里面,这里是一个 当我触摸时,它可能会激活,这将创建另一个具有不同属性的属性,这些属性会从屏幕上掉下来并在不到一秒钟的时间内销毁自己。这是我创建粒子效果的方式。我们可以将其称为“粒子” ,就像构造函数中的参数一样。 一切正常,直到我添加另一个main为止。现在,我同时在屏幕上有两个,如果我触摸最新的,它可以正常工作并启动粒子。 但是,如果我
问题内容: 假设你创建了一个新项目,并希望它使用某个第三方库,例如。你在哪里添加此JAR: 文件>>项目属性>>图书馆>> 编译时库 ; 要么 文件>>项目属性>>图书馆>> 运行时库 ; 要么 工具>>库(库管理器)>>库类路径;要么 工具>> Java平台(Java平台管理器) 所有这些对话框似乎都做相同的事情,但是我敢肯定它们都有正确的用法。在网上找不到好的“最佳实践”文章,并且“ NetB
问题内容: 我有两次弦乐时间 有没有简单的方法可以将这两个时间相加并获得一个新的时间 呢? 我想在客户端执行此操作,因此,如果可以避免使用任何日期数据库 问题答案: 请记住,您可以将小时/分钟/秒的整数值转换为单个整数,如下所示: 并转换回: 或者,您可以按以下步骤进行算术零碎: