基于C++实现五子棋AI算法思想

更多精彩,请点击上方蓝字关注我们!

  今天我想要分享一下我做五子棋AI的思路。因为在做这个之前,我没有接触过任何像这种类似的东西。通过这一次,我也算是有所了解,我的思路也是来自很多网络上的博客,看了很多,最终总结出了自己的这样一个。

    那我的五子棋是15*15的大小(一般也就是这样的一个大小)。我的AI算法要求每一次落子之后都要去计算每一个空暇的位置的“分值”,简单的说,我们需要一个存放棋子的数组,表示是否存放了棋子,还要一个计算每一个空格的数组来记录“分数”,这个分数是后期AI用来运算的基础,也是你AI难度控制的点。

     我现有的思路就是分两部分。首先是如果是玩家先落子,那么要求电脑AI随即在你落子的地方的任意一个方向,随机落子,这是第一步。接下来以后就正式进入到算法中去。

首先初始化你的分数数组,让他们全部为零。然后在每一次落子之后进行全盘的遍历,如果发现该处为空白,于是检查其四周八个方向(当然如果是边缘位置就相对修改,判断是否出了边界)。若在空白处,且发现在某一对角线方向发现有一个其他颜色的棋子,那么相对的给这个空白区域的分数数组加上一定的分值,然后继续往这个方向检测是否还有连续的同一颜色的棋子,若没有则检查其他方向或者检测下一个空白位置。若是还在同一方向上面找到了相同颜色的棋子,那么第二个棋子的出现,你可以给改空白处加上双倍的分值,表明这个空白位置更加重要。一次类推,继续检测。(PS:因为最终AI棋子落在什么地方,依靠的是最后遍历整个分数数组,然后根据分数的高低来进行判断落子落在哪里的,在下面讲)。

     经过上一遍的遍历,每一次落子都会使得分数数组得到一些变化,每一次都会导致AI判断的变化。在这个基础上,每一次落子还要进行一次对自己本身棋子颜色的一个遍历,判断自己的情况,同时加分加在分数数组之中,这样一来,电脑就会根据自己的棋子的情况以及玩家的落子情况进行判断,哪一个地方更加适合落子。

     因为我是第一次做AI,网络上搜到的一些思想一般也是这种类似的遍历思想。理解了以后写代码就比较方便。最后可能会有一些点的分数是相同的,所以还有设置一下随机落子。把分数相同的地点随机落子。

     个人感觉AI的强弱是根据你每一次给他增加分数的多少来确定的。这个我的AI有时候也会抽风,不过一般情况比较正常,可能运气也占了一部分,当初设计加分的时候其实没想那么多,现在却发现好像还不错。

      大家要多去实践练习,多改改分数可能就会出来不错的AI了,o(^▽^)o。

    下面贴上我的代码! 

void GameScene::Robot(int *x, int *y, int *Sum)
{ExWhile1 = true;if (*Sum == 1){while (ExWhile1){ChessOne(*x, *y);if (ch[*x][*y] == 2){ ExWhile1 = false; }}ch[*x][*y] = tp;   //记录这个点printpart(*x, *y, tp);     //打印出电脑AI第一次落子isTouch = true;tp++;tp = tp % 2;}else      //从第2步开始,使用评分系统{Findscore(*x, *y);}
}void GameScene::Findscore(int &x, int &y)   //查找评分最高的坐标
{srand((unsigned)time(NULL));int i, j, x1, x2, y1, y2, lx;int Max = 0;ChessScore();      //调用评分函数for (i = 0; i<15; i++){for (j = 0; j<15; j++){if (Score[i][j]>Max){Max = Score[i][j];  //获取所有点中,评分最高的x1 = i;y1 = j;}}}x2 = x1; y2 = y1;for (i = 0; i<15; i++)    //可能的话,有评分相同的多个点{for (j = 0; j<15; j++){if (Score[i][j] == Max&&i != x2&&j != y2)   //在这么多个相同分数的点中,随机找一个{lx = rand() % 10;if (lx<5){x2 = i, y2 = j;break;}}}}if (x2 != x1 || y2 != y1)   //棋盘上有2个最高分{lx = rand() % 10;    //随机一个if (lx>6){x = x1, y = y1;}else{x = x2, y = y2;}}else    //棋盘上只有一个最高分{x = x1, y = y1;}Max = 0;    //清空最大值ch[x][y] = tp;       //记录这个点printpart(x, y, tp);    //打印出电脑AI落子if (winerValue==2){isTouch = true;}tp++;tp = tp % 2;
}inline void GameScene::ChessOne(int &x, int &y)   //玩家走第1步时的落子
{int i, j;srand((unsigned)time(NULL));    //随机数随着时间的改变而改变for (i = 0; i<15; i++){for (j = 0; j<15; j++){if (ch[i][j] == 0)  //如果找到了玩家的棋子,在它的8个方的任意一点落子{int lx = rand() % 7;if (lx == 0){x = i + 1; y = j + 1;if (ch[x][y] == 2){ break; }}else if (lx == 1){x = i + 1; y = j - 1;if (ch[x][y] == 2){ break; }}else if (lx == 2){x = i - 1; y = j - 1;if (ch[x][y] == 2){ break; }}else if (lx == 3){x = i - 1; y = j + 1;if (ch[x][y] == 2){ break; }}else if (lx == 4){x = i - 1; y = j;    //上if (ch[x][y] == 2){ break; }}else if (lx == 5){x = i; y = j - 1;   //左if (ch[x][y] == 2){ break; }}else if (lx == 6){x = i; y = j + 1;   //右if (ch[x][y] == 2){ break; }}else{x = i + 1; y = j;    //下if (ch[x][y] == 2){ break; }}}}}
}void GameScene::ChessScore()
{int x, y, i, j, k;      //循环变量int number1 = 0, number2 = 0;   //number用来统计玩家或电脑棋子连成个数int empty = 0;    //empty用来统计空点个数memset(Score, 0, sizeof(Score));                    //把评分数组先清零for (x = 0; x<15; x++){for (y = 0; y<15; y++){if (ch[x][y] == 2)    //如果这个点为空  {for (i = -1; i <= 1; i++){for (j = -1; j <= 1; j++)   //判断8个方向  {if (i != 0 || j != 0)   //若是都为0的话,那不就是原坐标嘛{//对玩家落点评分for (k = 1; i <= 4; k++)   //循环4次{                        //这点没越界  且这点存在黑子(玩家)if (x + k*i >= 0 && x + k*i <= 14 &&  y + k*j >= 0 && y + k*j <= 14 &&  ch[x + k*i][y + k*j] == 0){  number1++;}else if (ch[x + k*i][y + k*j] == 2)     //这点是个空点,+1后退出{  empty++;  break;  }    else                    //否则是墙或者对方的棋子了  {  break;  }     }for (k = -1; k >= -4; k--)            //向它的相反方向判断  {                        //这点没越界  且这点存在黑子(玩家)if (x + k*i >= 0 && x + k*i <= 14 &&y + k*j >= 0 && y + k*j <= 14 &&  ch[x + k*i][y + k*j] == 0){  number1++;}else if (ch[x + k*i][y + k*j] == 2)     //这点是个空点,+1后退出{  empty++;  break;  }   else{  break;}}if (number2 == 1)   //2个棋子  {  Score[x][y] += 1;  }   else if (number1 == 2)   //3个棋子  {if (empty == 1)     {Score[x][y] += 5;   //有一个空点+5分 死3  }      else if (empty == 2)  {  Score[x][y] += 10;  //有两个空点+10分 活3}  }else if (number1 == 3)   //4个棋子  {if (empty == 1)    {  Score[x][y] += 20;  //有一个空点+20分 死4  }   else if (empty == 2){  Score[x][y] += 100;  //有2个空点+100分 活4}   }else if (number1 >= 4)   {  Score[x][y] += 1000;  //对方有5个棋子,分数要高点,先堵}   empty = 0;   //统计空点个数的变量清零  //对电脑落点评分for (k = 1; i <= 4; k++)   //循环4次{       //这点没越界  且这点存在白子(电脑)if (x + k*i >= 0 && x + k*i <= 14 &&  y + k*j >= 0 && y + k*j <= 14 &&  ch[x + k*i][y + k*j] == 1){  number2++;}else if (ch[x + k*i][y + k*j] == 2){  empty++; break;   //空点}  else{break;}}for (k = -1; k >= -4; k--)   //向它的相反方向判断  {if (x + k*i >= 0 && x + k*i <= 14 &&y + k*j >= 0 && y + k*j <= 14 &&  ch[x + k*i][y + k*j] == 1){  number2++;}else if (ch[x + k*i][y + k*j] == 2){  empty++; break;}else{  break;   //注释与上面玩家版相同}       }if (number2 == 0)    {  Score[x][y] += 1;    //1个棋子}  else if (number2 == 1)   {Score[x][y] += 2;    //2个棋子  }    else if (number2 == 2)   //3个棋子{if (empty == 1)    {  Score[x][y] += 8;  //死3}   else if (empty == 2)  {  Score[x][y] += 30;  //活3  }  }else if (number2 == 3)   //4个棋子{if (empty == 1)    {  Score[x][y] += 50;   //死4}   else if (empty == 2)  {  Score[x][y] += 200;   //活4}   }else if (number2 >= 4)  {  Score[x][y] += 10000;   //自己落在这点能形成5个,也就能胜利了,分数最高}  number1 = 0;     //清零,以便下次重新统计number2 = 0;                   empty = 0;}}}}}}
}

它,

不仅仅是一个码

扫码关注

C++资源免费送

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

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

相关文章

java代码初体验_第一次Java 8体验

java代码初体验像世界其他地方一样&#xff0c;我深深地爱上了Slack。 为什么&#xff1f; 原因很多&#xff0c;但主要的原因是它提供了一种围绕通讯而非工具真正构建SDLC流程的新方法。 您认为这些天哪个更常见&#xff0c;杂乱无章的机智团队在荒野中四处徘徊&#xff0c;尽…

C++学习 | 面试官:我们只想要这样的C++工程师

更多精彩&#xff0c;请点击上方蓝字关注我们&#xff01;最近好多同学来找就就说&#xff0c;我想找 xxx 方向的工作&#xff0c;我怎么学习呀&#xff1f;我也不知道这个岗位的招聘要求是啥&#xff1f;这个很正常哈&#xff0c;主要是咱们平时很少去关注企业的招聘要求&…

C语言开发推箱子游戏项目实战

C语言能做什么&#xff1f;相信学过C语言的各位都会有一个疑问&#xff0c;我们学了C语言到底可以做什么&#xff1f;或许有人会说做系统开发、驱动开发等等&#xff0c;但是这个感觉离我们好远&#xff0c;甚至于我们学C语言都仅仅是为了用程序做数学题&#xff0c;要不就是用…

dw php如何传递id参数6,[php] 我的微型论坛的简单教程[已完成]第6/8页

现在数据库中还没有数据&#xff0c;所以&#xff0c;我们运行首页&#xff0c;只显示“对不起&#xff0c;论坛尚在建设中……”。既然我们很希望看到结果&#xff0c;就往数据库中加几条数据吧&#xff01;&#xff01;当然&#xff0c;直接在MySQL客户端运行查询语句"i…

尺度不变性是指什么不变_不变性如何提供帮助

尺度不变性是指什么不变在最近的几篇文章中&#xff0c;包括“ Getters / Setters。 邪恶。 期。” &#xff0c; “对象应该是不可变的”和“依赖注入容器是代码污染者” &#xff0c;我普遍将所有可变对象标记为“ setter”&#xff08;以set开头的对象方法&#xff09;。 我…

java prototype是什么,java设计模式-原型模式(Prototype)

定义原型模式属于对象的创建模式。通过给出一个原型对象来指明所有创建的对象的类型&#xff0c;然后用复制这个原型对象的办法创建出更多同类型的对象。这就是原型模式的用意原型模式的结构原型模式要求对象实现同一个可以“克隆”自身的接口&#xff0c;遮掩个就可以通过赋值…

Windows 动态链接库 DLL 浅析

一、概念DLL&#xff1a;Dynamic Link Library&#xff0c;即动态链接库&#xff0c;这种库包含了可由多个程序同时使用的代码和数据。它是microsoft在windows操作系统中实现共享函数库概念的一种实现方式。其中windows中 一些作为DLL实现的文件有&#xff1a;ActiveX控件&…

图片大小 媒体大小自适应_自适应堆大小

图片大小 媒体大小自适应在改进测试平台以改进Plumbr GC问题检测器的同时 &#xff0c;我最终编写了一个小型测试用例&#xff0c;我认为这对于更广泛的读者来说可能很有趣。 我追求的目标是测试JVM在eden&#xff0c;survivor和Tenured空间之间如何分割堆方面的自适应性。 测…

如何优雅的激怒C/C++程序员

一、想知道怎么激怒C/C程序员及爱好者吗&#xff1f;那就来一起看看吧&#xff1a;C是一个编译很快&#xff0c;但运行很慢的语言。对此我不接受反驳&#xff0c;C我早就精通了。你问我需要多久才能精通&#xff0c;只要14天。前13天学C&#xff0c;最后1天看看面向对象就行。C…

java xml格式打包,maven项目打包xml没有被打包解决办法 ,mybatis的xml打包

在打maven包是遇见一个问题&#xff1a;打完包发现src/main/java中的的mapping没有被打包进去&#xff1b;原因主要是mapping目录里面的文件都是xml文件并不是.java文件&#xff0c;而maven打包默认的src/main/java的是Java文件&#xff0c;它不会打包里面的xml文件&#xff0c…

老司机找BUG指南,赶紧拿走。。

一.码畜&#xff1a;靠编译器帮自己查语法错误消灭笔误:编写适合程序员的键盘练习if (常量变量或表达式)使用goto接力超长的if,switch连续的if还是if elseif多个条件的组合:精心的排版多重括号的匹配条件编译各种const:不要纠结各种常量了&#xff0c;这个世界上唯一不变的就是…

红帽 jboss_红帽JBoss BRMS和BPMS富客户端框架展示了与GWT / Errai / UberFire和AngularJS的多语言集成...

红帽 jboss上周&#xff0c;我发布了一个博客&#xff0c;重点介绍了我的演示文稿&#xff0c;该演示文稿展示了我们在BRMS和BPMS平台内完成的工作&#xff0c;Drools和jBPM项目的产品化版本所产生的丰富客户端平台。 该演示文稿是所有屏幕截图和视频&#xff0c;您可以在此处找…

你知道自己适合做程序员吗?

哪有什么适不适合?编码本来就不是一件复杂的事情&#xff0c;日常工作不就是敲码、找Bug、 跟产品经理吵架、终于下班了、回家改BUG吗?今天看到了最奇葩的辞职理由&#xff1a;我觉得我数学不好&#xff0c;我不想做编程 我坐不住&#xff0c;我不想做编程 我性格太活泼&…

C/C++冷门知识点你知道多少呢?

在学习C语言的过程中&#xff0c;我们都自己总结了一些重要的知识点&#xff0c;但是有些不常用的冷门知识点可能就被我们选择性忘记了&#xff0c;你要不要来回一下呢&#xff1f;1&#xff1a;int跟signed int是一回事&#xff0c;short、long亦然&#xff0c;但char不是。ch…

判断人物眼型matlab,怎么判断眼型和脸型?

原标题&#xff1a;怎么判断眼型和脸型&#xff1f;为了判断脸部的形状&#xff0c;建议从面部类型诊断的长度和比例来衡量。我们只测量2个地方的长度和宽度。长度是从眉毛位置到嘴巴的长度&#xff0c;宽度是以嘴巴位置为中心到两个脸颊的宽度。脸型大致分为5种类型脸型大致分…

maven 打包编译_您是否真的想加快Maven的编译/打包速度? 那么takari生命周期插件就是答案。...

maven 打包编译像你们中的许多人一样&#xff0c;我正在使用多模块Maven项目 。 与现有的许多系统相比&#xff0c;它不是一个巨大的系统&#xff0c;它具有15个模块&#xff0c;3种不同的耳朵部署&#xff0c;带有属性文件的大量参数化以及大约10万行Java代码。 在开发高峰期&…

C/C++制作人机猜拳小游戏

C语言学完了之后总是感觉做不出什么有意思的东西&#xff0c;今天我们就来做一个好玩的小游戏试试。#include<stdio.h> #include<time.h> #include<conio.h> #include <windows.h>void win(int a, int b); void wait(); void softgame(); void lefttim…

r 数据框选子集_在带有组合框的值列表的下拉列表中显示显示属性的子集

r 数据框选子集组合框值列表&#xff08;inputComboboxListOfValues&#xff09;应该是使用LOV的非常流行的ADF Faces组件。 坦白说&#xff0c;这是我最喜欢的值列表方法。 在这篇简短的文章中&#xff0c;我将重点介绍ADF开发人员经常忽略的一项功能。 如果默认情况下定义了L…

1024到了,默默给自己点个赞!

只有程序员才能看懂的搞笑图片写了一大堆SQL语句&#xff0c;自己都不敢相信居然运行成功了找bug给客户演示&#xff0c;我的bug没被发现正则表达式返回预期效果正则表达式没有返回预期效果上线两天了&#xff0c;客户突然要改需求&#xff0c;我去年买了个表一口气写完代码&am…

input发送a.jax_Java EE 7 / JAX-RS 2.0:具有自定义HTTP标头的简单REST API身份验证和授权...

input发送a.jax在使用已可用的HTTP协议实施Web服务时&#xff0c;REST带来了很多便利。 通过仅通过指定的URL触发GET&#xff0c;POST和其他HTTP方法&#xff0c;您将确保通过REST服务的响应来完成某些工作。 但是&#xff0c;无论REST给开发人员带来了什么便利&#xff0c;安全…