Qt版本中国象棋开发(四)
内容:走法产生
中国象棋基础搜索AI,
极大值,极小值剪枝搜索,
静态估值函数
理论基础:
(一)人机博弈走法产生:
先遍历某一方的所有棋子,再遍历整个棋盘,得到每个棋子的所有走棋情况(效率不高,可以改进)
void SingleGame::getAllPossibleMove(QVector<Step *> &steps)
{
int min, max;
if(this->_bRedTurn)
{
min = , max = ;
}
else
{
min = , max = ;
} for(int i=min;i<max; i++)
{
if(this->_s[i]._dead) continue;
for(int row = ; row<=; ++row)
{
for(int col=; col<=; ++col)
{
int killid = this->getStoneId(row, col);
if(sameColor(i, killid)) continue; if(canMove(i, killid, row, col))
{
saveStep(i, killid, row, col, steps);
}
}
}
}
}
(二)棋局博弈树理论:
名词:对抗性搜索(Adversarial Search):敌对双方交替动作的搜索
博弈树:树的根部是棋局的初始局面,根的若干子节点是有甲的每一种可能走法生成的局面,
这些节点的子节点则是由乙的每一种可能走法生成的局面,如此交替直到棋局结束。

博弈树形象表示
基于博弈树的游戏:
计甲胜为WIN,乙胜为LOST,和局为DRAW;
轮到甲走时,甲选择通向WIN或DRAW的节点;换言之,turn 甲,选择所有子节点中最好的(对甲);
轮到乙走时,乙选择通向LOST或DRAW的节点;turn 乙,选择所有子节点中最差的(对甲);
(三)极大极小值算法
在上述博弈树基础上,令甲胜的局面值为1,乙胜的局面值为-1,和局的局面值为0;
轮到甲走时,选择子节点值最大的走法,轮到乙走时,选择子节点值最小的走法;
中间节点的值的确定:该局面轮到甲走,选择其子节点中的最大值,
该局面轮到乙走,选择其子节点中的最小值。
问题:实际的棋局不能简单的以1,-1,0三种状态表示,
需要加入评估棋局局面分数的估值函数,配合博弈树的搜索来确定局面分数。
实际解决方案:
估值函数:暂时以静态估值的方式形成估值函数(评估较为粗糙),
将棋局中的每个棋子按照重要程度赋一个值,
估值函数通过计算一方现存棋子的总分数来确定局面优劣情况。
代码示例:
int SingleGame::score()
{
enum TYPE{CHE, MA, PAO, BING, JIANG, SHI, XIANG};
int s[] = {, , , , , , , , , , , , , , , };
//int s[] = {1000,450,501,200,15000,200,200};
/*当头卒比重大*/
int scoreBlack = ;
int scoreRed = ;
/*计算红方分数*/
for(int i=; i<; ++i)
{
if(_s[i]._dead) continue;
//scoreRed += s[_s[i]._type];
scoreRed += s[i];
}
/*计算黑方分数*/
for(int i=; i<; ++i)
{
if(_s[i]._dead) continue;
//scoreBlack += s[_s[i]._type];
scoreBlack += s[i-];
}
return scoreBlack - scoreRed;
}
极大值极小值搜索方案:
深度优先搜索,优点是不必在内存中生成整个博弈树,可以将搜索过的部分从内存中去除,
采用递归形式,依次在min(int level,int curMin),max(int level,int curMax)之间递归调用,
剪枝以去除不必要的步数,在求极大值时,若下一步的值小于当前极大值,直接删除这一步,不予考虑,
在求极小值时,若下一步的值大于当前极小值,直接删除这一步,不予考虑。
在所有子节点中选出值最大的走法,就是电脑的最佳走法。
代码示例:
int SingleGame::getMinScore(int level, int curMin)
{
if(level == )
return score(); QVector<Step*> steps;
getAllPossibleMove(steps);
int minInAllMaxScore = ; while(steps.count())
{
Step* step = steps.last();
steps.removeLast(); fakeMove(step);
int maxScore = getMaxScore(level-, minInAllMaxScore);
unfakeMove(step);
delete step; if(maxScore <= curMin)
{
while(steps.count())
{
Step* step = steps.last();
steps.removeLast();
delete step;
}
return maxScore;
} if(maxScore < minInAllMaxScore)
{
minInAllMaxScore = maxScore;
}
}
return minInAllMaxScore;
}
int SingleGame::getMaxScore(int level, int curMax)
{
if(level == )
return score(); QVector<Step*> steps;
getAllPossibleMove(steps);
int maxInAllMinScore = -; while(steps.count())
{
Step* step = steps.last();
steps.removeLast(); fakeMove(step);
int minScore = getMinScore(level-, maxInAllMinScore);
unfakeMove(step);
delete step; if(minScore >= curMax)
{
while(steps.count())
{
Step* step = steps.last();
steps.removeLast();
delete step;
}
return minScore;
}
if(minScore > maxInAllMinScore)
{
maxInAllMinScore = minScore;
} }
return maxInAllMinScore;
}
代码示例:
Step* SingleGame::getBestMove()
{
Step* ret = NULL;
QVector<Step*> steps;
getAllPossibleMove(steps);
int maxInAllMinScore = -; while(steps.count())
{
Step* step = steps.last();
steps.removeLast(); fakeMove(step);
int minScore = getMinScore(this->_level-, maxInAllMinScore);
unfakeMove(step); if(minScore > maxInAllMinScore)
{
if(ret) delete ret; ret = step;
maxInAllMinScore = minScore;
}
else
{
delete step;
}
}
return ret;
}
一个简单的象棋AI,还有诸多优化之处,目前搜索深度最大为4,与初级玩家对弈输多胜少。
Qt版本中国象棋开发(四)的更多相关文章
- Qt版本中国象棋开发(一)
开发目的:实现象棋人机对战简单AI,网络对战,移植到android中. 开发平台:windows10 + Qt5.4 for android 开发语言:C++ 开发过程:1.棋盘绘制: 方法一:重写 ...
- Qt版本中国象棋开发(三)
实现功能:棋子初始化及走棋规则 棋子类: #ifndef STONE_H #define STONE_H #include <QString> class Stone { public: ...
- Qt版本中国象棋开发(二)
实现功能:棋盘绘制 核心函数: void paintEvent(QPaintEvent *); //QWidget自带的虚函数,重写后使用 QPainter 类来绘制图形 QPainter paint ...
- 基于QT的中国象棋
基于QT的中国象棋,可实现人人对战,人机对战,联网对战,可显示棋谱,可悔棋. 还有一些小毛病,我之后会找空把这个DEMO重新修改一下上传 链接:https://pan.baidu.com/s/1-eM ...
- Cocos2d-X开发中国象棋《四》设计游戏场景
设计完開始界面后就要设计游戏界面了 为了理清设计思路先看一张游戏界面效果图 游戏界面设计思路: 1.在窗体上放一张桌子 2.在桌子上放一个棋盘 3.在棋盘右边加入新局button,暂不实现详细的功能 ...
- Qt绘制中国象棋棋盘
这里主要用的是#include <QPainter>里面的paintEvent void Board::paintEvent(QPaintEvent*) { QPainter painte ...
- Cocos2d-X开发中国象棋《二》project文件概述
我在上一篇博客中介绍了象棋的功能.在接下来的博客中将向大家介绍使用Cocos2d-X怎样一步一步开发中国象棋 开发工具: Cocos2d-X2.2.3 VS2012 项目的文件夹: Classes:存 ...
- Python开发中国象棋实战(附源码)
Pygame 做的中国象棋,一直以来喜欢下象棋,写了 python 就拿来做一个试试,水平有限,电脑走法水平低,需要在下次版本中更新电脑走法,希望源码能帮助大家更好的学习 python.总共分 ...
- cocos2d-x游戏开发系列教程-中国象棋02-main函数和欢迎页面
之前两个博客讲述了象棋的规格和工程文件之后,我们继续深入的从代码开始学习cocos2dx 首先从程序入口main函数开始 main函数 int APIENTRY _tWinMain(HINSTANCE ...
随机推荐
- 再谈 PHP 未来之路
前段时间我写过一篇博文<phper:敢问路在何方>,分析了 PHPer 的困境以及 PHP 程序员的学习.进阶突破之路.同时我在知乎上也发过类似的提问.从大家的评论和回答看,大体分为以下几 ...
- inode block 软硬链接
inode block 软硬链接 1 inode 1.1 inode(索引节点)作用 (1)用于存储文件数据属性信息(2)用于存储数据指针信息 1.2 如何产生 格式化时,创建文件系统 1.3 如何查 ...
- 【Hadoop离线基础总结】MapReduce案例之自定义groupingComparator
MapReduce案例之自定义groupingComparator 求取Top 1的数据 需求 求出每一个订单中成交金额最大的一笔交易 订单id 商品id 成交金额 Order_0000005 Pdt ...
- repo使用总结—从入门到入门
文章目录 1 what's repo 2 官方文档 Repo 命令参考资料 help 帮助 init 初始化 sync 同步所有项目文件 upload 提交更改 diff download foral ...
- Altera特殊管脚的使用
- Python实现将网站域名解析为ip地址
起因 因为一些事情,需要将域名解析为ip地址,想到Python作为万能语言,就用Python来实现这个功能 代码 import socket url = 'shiyixirui.cn' res = s ...
- shrine
0x01 import flask import os app = flask.Flask(__name__) app.config['FLAG'] = os.environ.pop('FLAG') ...
- SpringBatch异常To use the default BatchConfigurer the context must contain no more thanone DataSource
SpringBoot整合SpringBatch项目,已将代码开源至github,访问地址:https://github.com/cmlbeliever/SpringBatch 欢迎star or fo ...
- mysql5.7 derived_merge=on 影响你的查询了吗?
衍生表的优化:合并 | 具化 一.mysql优化器对于衍生表的优化处理可以从两方面进行: 将衍生表合并到外部查询 将衍生表具化为内部临时表 1.示例 1: SELECT * FROM (SELECT ...
- Redis学习笔记(十) 客户端
Redis服务器是典型的一对多服务器程序:一个服务器可以与多个客户端建立网络连接,每个客户端可以向服务器发送命令请求,而服务器则接收并处理客户端发送的命令请求,并向客户端返回命令回复. 通过使用由I/ ...