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 ...
随机推荐
- 曹工谈并发:Synchronized升级为重量级锁后,靠什么 API 来阻塞自己
背景 因为想知道java中的关键字,对应的操作系统级别的api是啥,本来打算整理几个我知道的出来,但是,尴尬的是,我发现java里最重要的synchronized关键字,我就不知道它对应的api是什么 ...
- 细说 PEP 468: Preserving Keyword Argument Order
细说 PEP 468: Preserving Keyword Argument Order Python 3.6.0 版本对字典做了优化,新的字典速度更快,占用内存更少,非常神奇.从网上找了资料来看, ...
- IoTClient开发6 - S7-200SmarTcp协议客户端实现
环境和工具 服务端电脑IP:192.168.1.130 客户端电脑IP:192.168.1.120 1.在服务端电脑运行IoTClientTool 2.运行Wireshark 3.在客户端电脑运行Io ...
- Linux查看redis占用内存的方法
redis-cli auth 密码info # Memory used_memory:13490096 //数据占用了多少内存(字节) used_memory_human:12.87M //数据占用了 ...
- 实用,小物体检测的有监督特征级超分辨方法 | ICCV 2019
论文提出新的特征级超分辨方法用于提升检测网络的小物体检测性能,该方法适用于带ROI池化的目标检测算法.在VOC和COCO上的小物体检测最大有5~6%mAP提升,在Tsinghua-Tencent 10 ...
- Coursera课程笔记----P4E.Capstone----Week 2&3
Building a Search Engine(week 2&3) Search Engine Architecture Web Crawling Index Building Search ...
- 给大家发个Python和Django的福利吧,不要钱的那种~~~
前言一: 这篇是一个发放福利的文章,但是发放之前,我还是想跟大家聊聊我为什么要发这样的福利. 我第一份工作是做的IT桌面支持,日常工作就是给同事修修电脑.装装软件.开通账号.维护内部系统之类的基础工作 ...
- 【FreeRTOS学习02】源码结构/数据类型/命名规则总结
个人不是很喜欢FreeRTOS的编程风格,但是没办法,白嫖人家的东西,只能忍了,这里先简单总结一下: 相关文章 [FreeRTOS实战汇总]小白博主的RTOS学习实战快速进阶之路(持续更新) 文章目录 ...
- 存储过程——公用表表达式(CTE)
目录 0. 背景说明 1. 定义及语法细节 1.1 基本定义 1.2 基本语法 1.3 多个CTE同时声明 1.4 CTE嵌套使用 2. CTE递归查询 2.1 简介 2.2 准备工作 2.3 计算每 ...
- python基础学习笔记(纸质)
大一的时候学的python做的笔记.