AI五子棋——超强改进版本

AI五子棋的改进版本来啦~~
我们发现,原版的AI五子棋如果调成4的话,非常之慢!!下面给出原版的链接

AI五子棋(原版本)

因此我对其进行了改进,由于正常人下五子棋不会东下一颗棋,西下一颗棋。
因此我们可以大幅度缩小搜索的范围,只要搜索已经下了的棋子的周围就可以了(2×2或3×3)。
下面的程序会是2×2的4,尽管还是有点慢,但相比原程序快很多。

另外,关于人机强度、人机耗时的修改,我放在原版的链接里了

我们可以做一个对比:让这两个程序同时运行4,会发现:改进后的程序用了1:17下了一步,但是原版的用了3:04(4GB内存,i3cpu情况下)

#include<iostream>
#include<vector>
#include<array>
#include<fstream>
#include<algorithm>
#include <bits/stdc++.h>
#include <windows.h>
using namespace std;
int minx=14,miny=14,maxx=0,maxy=0;
void color(int x) {switch(x) {case  1:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_RED  );break;case  2:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_BLUE );break;case  3:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_GREEN);break;case  4:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_RED  |FOREGROUND_BLUE );break;case  5:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_RED  |FOREGROUND_GREEN);break;case  6:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_BLUE |FOREGROUND_GREEN);break;case  7:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_GREEN|FOREGROUND_BLUE |FOREGROUND_RED);break;default:SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY |FOREGROUND_GREEN|FOREGROUND_BLUE |FOREGROUND_RED);break;}
}
class GameTree {public:class Node {public:enum State :uint8_t/*节省内存*/ { SPACE, BLACK, WHITE };private:friend class GameTree;static const uint8_t BOARDSIZE = 15;int value;//叶节点表示估价函数的结果,MAX节点表示α值,MIN节点表示β值int evaluateValue;//估价函数计算的结果,用于优化unsigned short depth;//深度uint8_t lastX, lastY;//上一次落子的xy坐标Node* father;//父亲节点std::vector<Node*> children;//子节点State **board;//棋盘int Evaluate()const { //估价函数static const auto EvaluateSome = [](State **board, uint8_t beginX, uint8_t endX, uint8_t beginY, uint8_t endY) {static const auto EvaluateList = [](const std::array<State, 5>& v) { //假定自己是白方//判断颜色并记录棋子个数State lastColor = SPACE;uint8_t bitList = 0;//将棋链以二进制形式表示,如01101for (State i : v) {if (i != lastColor) {if (lastColor == SPACE)//遇到的第一个棋子lastColor = i;else//有不同颜色的棋子return 0;}if (i != SPACE)bitList = bitList * 2 + 1;}int result = 0;switch (bitList) {case 0://00000result = 0;break;case 1://00001case 2://00010case 4://00100case 8://01000case 16://10000result = 5;break;case 3://00011case 24://11000result = 80;break;case 6://00110case 12://01100result = 100;break;case 10://01010result = 80;break;case 5://00101case 20://10100result = 60;break;case 9://01001case 18://10010result = 20;break;case 17://10001result = 10;break;case 7://00111case 28://11100result = 800;break;case 14://01110result = 1000;break;case 13://01101case 26://11010case 11://01011case 22://10110result = 800;break;case 19://10011case 21://10101case 25://11001result = 600;break;case 15://01111case 30://11110result = 10000;break;case 29://11101case 23://10111result = 8000;break;case 27://11011result = 6000;break;case 31://11111return lastColor == WHITE ? INT_MAX : INT_MIN;}return lastColor == WHITE ? result : -result;//对手返回负值,我方返回正值};int result = 0;for (uint8_t i = beginX; i < endX; i++) { //分别从四个方向判断for (uint8_t j = beginY; j < endY; j++) {if (j + 4 < endY) {std::array<State, 5>v;for (uint8_t k = 0; k < 5; k++)v[k] = board[i][j + k];const int t = EvaluateList(v);if (t == INT_MAX || t == INT_MIN)//决出胜负直接返回return t;result += t;}if (i + 4 < endX) {std::array<State, 5>v;for (uint8_t k = 0; k < 5; k++)v[k] = board[i + k][j];const int t = EvaluateList(v);if (t == INT_MAX || t == INT_MIN)//决出胜负直接返回return t;result += t;}if (i + 4 < endX && j + 4 < endY) {std::array<State, 5>v;for (uint8_t k = 0; k < 5; k++)v[k] = board[i + k][j + k];const int t = EvaluateList(v);if (t == INT_MAX || t == INT_MIN)//决出胜负直接返回return t;result += t;}if (i + 4 < endX && j >= 4) {std::array<State, 5>v;for (uint8_t k = 0; k < 5; k++)v[k] = board[i + k][j - k];const int t = EvaluateList(v);if (t == INT_MAX || t == INT_MIN)//决出胜负直接返回return t;result += t;}}}return result;};uint8_t beginX, endX, beginY, endY;if (lastX <= 5)beginX = 0;elsebeginX = lastX - 5;endX = lastX + 5;if (endX > BOARDSIZE)endX = BOARDSIZE;if (lastY <= 5)beginY = 0;elsebeginY = lastY - 5;endY = lastY + 5;if (endY > BOARDSIZE)endY = BOARDSIZE;const int t = EvaluateSome(board, beginX, endX, beginY, endY);if (t == INT_MAX || t == INT_MIN)//决出胜负直接返回return t;return  t - EvaluateSome(father->board, beginX, endX, beginY, endY) + father->evaluateValue;}public://非根节点的构造函数Node(Node* nf, uint8_t x, uint8_t y) :father(nf), lastX(x), lastY(y), depth(nf->depth + 1), value(0), board(new State* [BOARDSIZE]) {father->children.push_back(this);for (int i = 0; i < BOARDSIZE; ++i) {board[i] = new State[BOARDSIZE];memcpy(board[i], father->board[i], BOARDSIZE * sizeof(State));}board[lastX][lastY] = IsMaxNode() ? BLACK : WHITE;evaluateValue = Evaluate();for (int i = 0; i < BOARDSIZE; ++i) {delete[] board[i];}delete[] board;board = nullptr;}//根节点的构造函数Node(int _depth, uint8_t x, uint8_t y) :father(nullptr), depth(_depth), lastX(x), lastY(y), value(0),evaluateValue(0),board(new State*[BOARDSIZE]) {for (int i = 0; i < BOARDSIZE; ++i) {board[i] = new State[BOARDSIZE];memset(board[i], 0, BOARDSIZE * sizeof(State));}board[x][y] = IsMaxNode() ? BLACK : WHITE;}inline int GetEvaluateValue()const {return evaluateValue;}inline bool IsMaxNode()const { //默认计算机后手return depth & 1u;//相当于depth%2}void Search(unsigned short _depth) {if (_depth == 0 || this->evaluateValue == INT_MAX || this->evaluateValue == INT_MIN) {this->value = this->evaluateValue;return;}bool created = false;//记录是否new出新的Node,如果没有就不用排序了。if (!board) { //不是根节点board = new State * [BOARDSIZE];for (int i = 0; i < BOARDSIZE; ++i) {board[i] = new State[BOARDSIZE];memcpy(board[i], father->board[i], BOARDSIZE * sizeof(State));}board[lastX][lastY] = IsMaxNode() ? BLACK : WHITE;}for (int i = max(minx-2,0); i < min(maxx+2,14); i++) {for (int j = max(miny-2,0); j < min(maxy+2,14); j++) {if (!board[i][j]) {bool flag = false;if (_depth >= 2) { //若剩余深度小于2,则下一层肯定没有搜索过for (Node* child : this->children) {if (child->lastX == i && child->lastY == j) { //已经被搜索过flag = true;break;}}}if (!flag) {new Node(this, i, j);minx=min(minx,i);maxx=max(maxx,i);miny=min(miny,j);maxy=max(maxy,j);created = true;}}}}if (IsMaxNode()) {this->value = INT_MIN;if (created) {std::sort(this->children.begin(), this->children.end(), [](Node* a, Node* b) {return a->GetEvaluateValue() > b->GetEvaluateValue();});//按照估价从大到小排序,增加剪枝的概率}} else {this->value = INT_MAX;if (created) {std::sort(this->children.begin(), this->children.end(), [](Node* a, Node* b) {return a->GetEvaluateValue() < b->GetEvaluateValue();});//按照估价从小到大排序,增加剪枝的概率}}auto ReleaseMemory = [this] {if (father) { //不是根节点for (int i = 0; i < BOARDSIZE; ++i) {delete[] board[i];}delete[] board;board = nullptr;}};for (Node* child : this->children) {child->Search(_depth - 1);//α - β 剪枝if (IsMaxNode()) {if (child->value > this->value) {this->value = child->value;if (this->father && this->value >= this->father->value && this != this->father->children.front()) {ReleaseMemory();return;//剪枝}}} else { //MIN节点if (child->value < this->value) {this->value = child->value;if (this->father && this->value <= this->father->value && this != this->father->children.front()) {ReleaseMemory();return;//剪枝}}}}ReleaseMemory();}void DeleteAllButThis() {if (!father->board)//父节点不是根节点throw std::invalid_argument("this->father必须是根节点!");for (Node* n : father->children) {if (n != this)delete n;}board = new State * [BOARDSIZE];for (int i = 0; i < BOARDSIZE; ++i) {board[i] = new State[BOARDSIZE];memcpy(board[i], father->board[i], BOARDSIZE * sizeof(State));delete[] father->board[i];}delete[] father->board;board[lastX][lastY] = IsMaxNode() ? BLACK : WHITE;free(father);//避免调用析构函数father = nullptr;}inline bool IsFull()const {return depth == BOARDSIZE * BOARDSIZE;}inline State GetWinner()const {if (this->value == INT_MAX) {return Node::WHITE;} else if (this->value == INT_MIN) {return Node::BLACK;}return Node::SPACE;}~Node() {if (board) {for (int i = 0; i < BOARDSIZE; ++i) {delete[] board[i];}delete[] board;}for (Node* n : children) {delete n;}}};private:Node* root;const unsigned short maxDepth;public:GameTree(unsigned short _maxDepth) : root(nullptr), maxDepth(_maxDepth) {if (_maxDepth < 2)throw std::invalid_argument("最大层数必须大于等于2!");}std::pair<uint8_t, uint8_t>AIGetNextPos(uint8_t x, uint8_t y) {if (root) {for (Node* node : root->children) { //进入用户选择的分支if (node->lastX == x && node->lastY == y) {node->DeleteAllButThis();root = node;break;}}} else { //第一次开局root = new Node(1, x, y);}root->Search(maxDepth);if (root->IsFull())return std::make_pair(19, 19);for (Node* node : root->children) { //选择分值最大的if (node->value == root->value) {node->DeleteAllButThis();root = node;break;}}return std::make_pair(root->lastX, root->lastY);}Node::State GetWinner()const {return root->GetWinner();}void Run() {while (1) {int x, y;do {color(7);std::cout << "输入x,y坐标";color(3);std::cin >> y >> x;y-=1;x-=1;} while (x < 0 || y < 0 || x >= 15 || y >= 15 || (root && root->board[x][y] != Node::SPACE));minx=min(minx,y);maxx=max(maxx,y);miny=min(miny,x);maxy=max(maxy,x);if (root) {for (Node* node : root->children) { //进入用户选择的分支if (node->lastX == x && node->lastY == y) {node->DeleteAllButThis();root = node;break;}}} else { //第一次开局root = new Node(1, x, y);}system("cls");for (int i = 0; i <= 15; i++) {for (int j = 0; j <= 15; j++) {color(6);if (i==0 && j<10)cout <<" " <<j <<" ";else if (i==0 && j>=10)cout <<j <<" ";else if (j==0 && i<10)cout <<" " <<i <<" ";else if (j==0 && i>=10)cout <<i <<" ";else if (root->board[i-1][j-1] == Node::SPACE) {color(5);std::cout << "十 ";} else if (root->board[i-1][j-1] == Node::BLACK) {color(2);std::cout << "○ ";} else {color(7);std::cout << "○ ";}}std::cout << '\n';}root->Search(maxDepth);if (root->value == INT_MAX) {std::cout << "电脑胜利!";break;} else if (root->value == INT_MIN) {std::cout << "玩家胜利!";break;} else if (root->IsFull()) { //不能用root->value==0判断平局,因为平局一定为0,但为0不一定平局std::cout << "平局!";break;}for (Node* node : root->children) { //选择分值最大的if (node->value == root->value) {node->DeleteAllButThis();root = node;break;}}system("cls");for (int i = 0; i <= 15; i++) {for (int j = 0; j <= 15; j++) {color(6);if (i==0 && j<10)cout <<" " <<j <<" ";else if (i==0 && j>=10)cout <<j <<" ";else if (j==0 && i<10)cout <<" " <<i <<" ";else if (j==0 && i>=10)cout <<i <<" ";else if (root->board[i-1][j-1] == Node::SPACE) {color(5);std::cout << "十 ";} else if (root->board[i-1][j-1] == Node::BLACK) {color(2);std::cout << "○ ";} else {color(7);std::cout << "○ ";}}std::cout << '\n';}if (root->value == INT_MAX) {std::cout << "电脑胜利!";break;} else if (root->value == INT_MIN) {std::cout << "玩家胜利!";break;} else if (root->IsFull()) { //不能用root->value==0判断平局,因为平局一定为0,但为0不一定平局std::cout << "平局!";break;}}}~GameTree() {delete root;}
};
int main() {cout <<"\n\n\n\n\n\n                           ";color(6);Sleep(1000);cout <<"游";Sleep(1000);cout <<"戏";Sleep(1000);cout <<"开";Sleep(1000);cout <<"始";Sleep(1000);cout <<"!";Sleep(2000);system("cls");for (int i = 0; i <= 15; i++) {for (int j = 0; j <= 15; j++) {color(6);if (i==0 && j<10)cout <<" " <<j <<" ";else if (i==0 && j>=10)cout <<j <<" ";else if (j==0 && i<10)cout <<" " <<i <<" ";else if (j==0 && i>=10)cout <<i <<" ";else {color(5);std::cout << "十 ";}}std::cout << '\n';}GameTree(4).Run();	//较为智能的模式,可改为3、2(不智能模式),以及5(超级智能模式,但是特别慢) return 0; 
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/28785.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Flutter Dart语言(05)异步

0 说明 该系列教程主要是为有一定语言基础 C/C的程序员&#xff0c;快速学习一门新语言所采用的方法&#xff0c;属于在C/C基础上扩展新语言的模式。 1 async和await 在Dart语言中&#xff0c;虽然没有像其他语言&#xff08;如Java、C、Python&#xff09;中的传统多线程概…

Java怎么手动将对象注入到springboot

在Java中&#xff0c;可以使用Spring的ApplicationContext来手动将对象注入到Spring Boot中。 1. 首先&#xff0c;确保你已经在Spring Boot应用程序中引入了Spring的依赖&#xff0c;比如 spring-boot-starter 。 2. 在你的类中注入ApplicationContext对象&#xff1a; Autowi…

谷歌广告(Google ads)如何投放?新手必看的超全教程

Google是公认的全球最大的搜索引擎&#xff0c;同时&#xff0c;Google还通过旗下的 YouTube、Gmail、Google Play、Android等产品&#xff0c;汇集了海量的海外用户。对于跨境出海商家来说&#xff0c;谷歌广告是提高销售额、提高产品流量、拓展全球市场的重要推广渠道。 那么…

一文走进时序数据库性能测试工具 TSBS

一、背景 在物联网、车联网等时序数据场景中&#xff0c;数据的高速写入能力至关重要&#xff0c;会对产品方案的可用性、可靠性和扩展性产生影响。 以物联网为例&#xff0c;当面临千万甚至上亿设备、平均每个设备采集几十个到几百个指标时&#xff0c;每秒生成的数据将达到…

Docker+Consul+Registrator 实现服务注册与发现

第四阶段 时 间&#xff1a;2023年8月8日 参加人&#xff1a;全班人员 内 容&#xff1a; DockerConsulRegistrator 实现服务注册与发现 目录 一、服务注册中心引言 CAP理论是分布式架构中重要理论&#xff1a; 二、服务注册中心软件 &#xff08;一&#xff09;Zoo…

ELK 将数据流转换回常规索引

ELK 将数据流转换回常规索引 现象&#xff1a;创建索引模板是打开了数据流&#xff0c;导致不能创建常规索引&#xff0c;并且手动修改、删除索引模板失败 "reason" : "composable template [logs_template] with index patterns [new-pattern*], priority [2…

中国剩余定理的同态性质(CRT变换的同态性)

1、中国剩余定理简介&#xff08;Chinese Remainder Theory&#xff0c;CRT&#xff09; 中国剩余定理是关于求解一元线性同余方程组的方法&#xff0c;用形式化的描述就是&#xff1a; m 1 , m 2 , m n m_1,m_2,m_n m1​,m2​,mn​是两两互素的n个整数&#xff0c;有下面的同…

MQTT 订阅接收消息 mosquitto 方式

1 说明 采用 mosquitto 库&#xff0c;实现订阅主题&#xff0c;并接收消息。其中服务器有做限制&#xff0c;需要对应的 cilent id &#xff0c;cafile 、certfile 、keyfile 等配置2 环境 采用ubuntu 直接编译调试 安装mosquitto 库 sudo apt install libmosquitto-dev su…

chatGLM 本地部署(windows+linux)

chatGLM算是个相对友好的模型&#xff0c;支持中英文双语的对话交流&#xff0c;清华出的 我的教程无需特别的网络设置&#xff0c;不过部分情况因为国内网络速度慢&#xff0c;需要反复重复 chatGLM github地址 一、硬件需求 N卡8G显存以上&#xff0c;最好16G以上&#xff…

【PHP的设计模式】

PHP的设计模式 一、策略模式二、工厂模式三、单例模式四、注册模式五、适配器模式六、观察者模式 一、策略模式 策略模式是对象的行为模式&#xff0c;用意是对一组算法的封装。动态的选择需要的算法并使用。 策略模式指的是程序中涉及决策控制的一种模式。策略模式功能非常强…

vue 使用vue-video-player加载视频

vue 使用vue-video-player加载视频 安装 npm install vue-video-player --savemain.js 引入 import VideoPlayer from "vue-video-player" import "video.js/dist/video-js.css" import "vue-video-player/src/custom-theme.css" import &quo…

SuperMap GIS基础产品桌面GIS FAQ集锦(5)

SuperMap GIS基础产品桌面GIS FAQ集锦&#xff08;5&#xff09; 【iDesktop】【iDesktopX】态势推演怎么使用&#xff1f; 【解决办法】 1、要先新建一个CAD数据集&#xff0c;然后将标绘加到CAD数据集中。 2、再使用态势推演管理器&#xff0c;右键新建分组。 3、选中场景中的…

利用ffmpeg分析视频流

ffprobe -show_packets -i "rtsp://192.168.61.46:8554/live?channel0&type0"&#xff1a;该命令用于显示 RTSP 流中的数据包信息&#xff0c;例如时间戳、大小、持续时间等。 ffprobe -i "rtsp://192.168.61.46:8554/live?channel0&type0"&…

【算法|数组】手撕经典二分法

算法|数组——二分查找 文章目录 算法|数组——二分查找引言二分查找左闭右闭写法左闭右开写法 总结 引言 首先学习这个算法之前需要了解数组知识&#xff1a;数组。 大概介绍以下&#xff1a; 数组是存储在连续内存空间上的相同类型数据的集合。数组下标都是从0开始。数组在…

数学建模—多元线性回归分析(+lasso回归的操作)

第一部分&#xff1a;回归分析的介绍 定义&#xff1a;回归分析是数据分析中最基础也是最重要的分析工具&#xff0c;绝大多数的数据分析问题&#xff0c;都可以使用回归的思想来解决。回归分析的人数就是&#xff0c;通过研究自变量X和因变量Y的相关关系&#xff0c;尝试去解释…

UML 用例图,类图,时序图,活动图

UML之用例图&#xff0c;类图&#xff0c;时序图&#xff0c;活动图_用例图 时序图_siyan985的博客-CSDN博客 https://www.cnblogs.com/GumpYan/p/14734357.html 用例图与类图 - 简书

webshell链接工具-Godzilla(哥斯拉)

项目地址 https://github.com/BeichenDream/Godzilla

势不可挡!新能源车型L2搭载率破50%,TOP20品牌数据出炉

中国乘用车市场正在走出去年的阴霾。 机构公开数据显示&#xff0c;2023年上半年中国乘用车市场产量1100.77万辆&#xff0c;同比增长7.5%&#xff1b;终端销量959.08万辆&#xff0c;同比增长4.7%&#xff1b;乘用车出口152.36万辆&#xff0c;同比增长87.6%。 在实际交付量…

JMeter启动时常见的错误

很多小伙伴在学工具这一块时&#xff0c;安装也是很吃力的一个问题&#xff0c;之前记得有说过怎么安装jmeter这个工具。那么你要启动jmeter的时候&#xff0c;一些粉丝就会碰到如下几个问题。 1.解压下载好的jmeter安装&#xff0c;Windows 平台&#xff0c;双击 jmeter/bin …

中间件RabbitMQ消息队列介绍

1. MQ的相关概念 1.1 什么是MQ MQ&#xff08;message queue&#xff09;&#xff0c;从字面意思上看&#xff0c;本质是个队列&#xff0c;FIFO先入先出&#xff0c;只不过队列中存放的内容是message而已&#xff0c;还是一种跨进程的通信机制&#xff0c;用于上下游传递消息…