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 ...
随机推荐
- RF(三层封装设计)
一.用例分层思想 元素层:需要导入 Selenium2Library 库 包含所有的元素定位 流程层:需要导入 元素层.txt 资源 封装用例流程 案例层:需要导入 流程层.txt 资源 输出用例,传 ...
- 9) drf JWT 认证 签发与校验token 多方式登陆 自定义认证规则反爬 admin密文显示
一 .认证方法比较 1.认证规则图 django 前后端不分离 csrf认证 drf 前后端分离 禁用csrf 2. 认证规则演变图 数据库session认证:低效 缓存认证:高效 jwt认证:高效 ...
- 写给Android 混淆小白的快速混淆方法
为啥子要混淆 简单来说,Android 进行ProGuard,可以起到压缩,混淆,预检,优化的功能,虽然不能说更安全但还是一个不容忽视的环节. 开始混淆第一步 首先在build.gradle 中将混淆 ...
- VSCode 安装 React 项目
1 下载nodejs 安装 (此时npm 和 node环境都已经装好) 2 安装淘宝镜像 npm install -g cnpm --registry=https://registry.npm.tao ...
- Python基础01 集合
初始化 # python3 # coding = utf-8 mylist = [] for item in range(10): mylist.append(item * 10 + 3) myset ...
- 自定义Element父子不关联的穿梭树
Element自身是有一个Transfer穿梭框组件的,这个组件是穿梭框结合checkbox复选框来实现的,功能比较单一,自己想实现这个功能也是很简单的,只是在项目开发中,项目排期紧,没有闲功夫来实现 ...
- css实现文字相对于图片垂直居中
一 要实现的样式,文字在图片的垂直居中位置 二 实现的代码: <style> .flag{ position: absolute; bottom: 0; width: 23rem; hei ...
- python学习之 %s %d 以及%变量名的含义
%age是对前面age变量的引用,%d是将这个变量名为age的值加到其中,但是如果变量值为字符串类型,则这里应该写成%s 也就是说当变量值为数值类型,而且必须是整型类型 应该使用%d 当变量值为字符串 ...
- 一篇文章解决MongoDB的所有问题
目录 一.MongoDB相关概念 1.1 业务应用场景 1.1.1 而MongoDB可应对"三高"需求· 1.1.2 什么时候选择MongoDB? 1.1.3 如果用mysql? ...
- Python 简明教程 --- 0,前言
微信公众号:码农充电站pro 个人主页:https://codeshellme.github.io Life is short, you need Python! -- Bruce Eckel 0,关 ...