《黑白棋》也叫翻转棋或者奥赛罗,其游戏过程是相互翻转对方的棋子,最后以棋盘上谁的棋子多来判断胜负。虽然规则简单,但是变化复杂,是典型的易学难精,奥妙无穷,不信您就试试看吧!
游戏规则
轮到一方下棋时,必须把棋下在与对方棋子相邻的空位上,要求所下的棋子和原有的已方棋子夹住对方的至少一个棋子(横竖斜夹均可),然后把被夹住的子变成己方的颜色(也叫吃子)。下棋过程中,任何棋子既不会从棋盘上拿走,也不会从一个格子移到另一个格子。在轮到一方下棋时,如果该方无子可吃,就必须停一步,让对方继续下棋,这种情况叫pass;而当一方在有子可吃时,即使想pass也不允许pass 。
今天我们就用C语言来实现它!
由于项目包含很多图片,因此这里给出整个项目的压缩包:https://codebus.cn/f/a/0/0/1/reversi.zip
以下是项目的全部源代码:
#include #include #include /*******************************定义全局变量*****************************/ char map[8][8]; //棋盘 IMAGE img[6]; //保存图片 int black, white; //双方的棋子数 char today; //当前轮到谁走 /**********************************定义函数*****************************/ void load(void) //加载图片 { loadimage(&img[0], "图片黑空.jpg"); loadimage(&img[1], "图片白空.jpg"); loadimage(&img[2], "图片黑子黑空.jpg"); loadimage(&img[3], "图片黑子白空.jpg"); loadimage(&img[4], "图片白子黑空.jpg"); loadimage(&img[5], "图片白子白空.jpg"); } void print(void) //画棋盘 { int x, y; black = white = 0; for(x = 0; x < 8; x++) for(y = 0; y < 8; y++) switch(map[x][y]) { case 0: if((x + y) % 2) putimage(60 * y, 60 * x, &img[0]); else putimage(60 * y, 60 * x, &img[1]); break; case 'B': if((x + y) % 2) putimage(60 * y, 60 * x, &img[2]); else putimage(60 * y, 60 * x, &img[3]); black++; break; case 'W': if((x + y) % 2) putimage(60 * y, 60 * x, &img[4]); else putimage(60 * y, 60 * x, &img[5]); white++; break; } } inline void print1(void) //画当前谁走的 { setcolor(WHITE); bar(530, 60, 590, 120); bar(530, 360, 590, 420); if(today == 'B') putimage(530, 60, &img[3]); else putimage(530, 360, &img[4]); } void draw(int x, int y, char a) //下当前子 { char b = ((a == 'B') ? 'W' : 'B'); //敌方子 int x1, y1, x2, y2; bool sign = false; //是否越过敌方子 for(x1 = x - 1; x1 >= 0 && map[x1][y]; x1--) //判断上方 { if(map[x1][y] == b) sign = true; else { if(sign) { for(x2 = x - 1; x2 > x1; x2--) //判断下方 map[x2][y] = a; } break; } } sign = false; for(x1 = x + 1; x1 < 8 && map[x1][y]; x1++) //判断右方 { if(map[x1][y] == b) sign = true; else { if(sign) { for(x2 = x + 1; x2 < x1; x2++) map[x2][y] = a; } break; } } sign = false; for(y1 = y - 1; y1 >= 0 && map[x][y1]; y1--) //判断左方 { if(map[x][y1] == b) sign = true; else { if(sign) { for(y2 = y - 1; y2 > y1; y2--) map[x][y2] = a; } break; } } sign = false; for(y1 = y + 1; y1 < 8 && map[x][y1]; y1++) //判断右方 { if(map[x][y1] == b) sign = true; else { if(sign) { for(y2 = y + 1; y2 < y1; y2++) map[x][y2] = a; } break; } } sign = false; for(x1 = x - 1, y1 = y - 1; x1 >= 0 && y1 >= 0 && map[x1][y1]; x1--, y1--) //左上方 { if(map[x1][y1] == b) sign = true; else { if(sign) { for(x2 = x - 1, y2 = y - 1; x2 > x1 && y2 > y1; x2--, y2--) map[x2][y2] = a; } break; } } sign = false; for(x1 = x + 1, y1 = y + 1; x1 < 8 && y1 < 8 && map[x1][y1]; x1++, y1++) //右下方 { if(map[x1][y1] == b) sign = true; else { if(sign) { for(x2 = x + 1, y2 = y + 1; x2 < x1 && y2 < y1; x2++, y2++) map[x2][y2] = a; } break; } } sign = false; for(x1 = x + 1, y1 = y - 1; x1 < 8 && y1 >= 0 && map[x1][y1]; x1++, y1--) //左下方 { if(map[x1][y1] == b) sign = true; else { if(sign) { for(x2 = x + 1, y2 = y - 1; x2 < x1 && y2 > y1; x2++, y2--) map[x2][y2] = a; } break; } } sign = false; for(x1 = x - 1, y1 = y + 1; x1 >= 0 && y1 < 8 && map[x1][y1]; x1--, y1++) //右上方 { if(map[x1][y1] == b) sign = true; else { if(sign) { for(x2 = x - 1, y2 = y + 1; x2 > x1 && y2 < y1; x2--, y2++) map[x2][y2] = a; } break; } } map[x][y] = a; print(); } bool judge(int x, int y, char a) //判断当前是否可以落下,同draw函数 { char b = ((a == 'B') ? 'W' : 'B'); int x1, y1; bool sign = false, sign1 = false; //sign1判断是否有效 if(map[x][y]) //如果当前不是空的返回假值 return false; for(x1 = x - 1; x1 >= 0 && map[x1][y]; x1--) { if(map[x1][y] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(x1 = x + 1; x1 < 8 && map[x1][y]; x1++) { if(map[x1][y] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(y1 = y - 1; y1 >= 0 && map[x][y1]; y1--) { if(map[x][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(y1 = y + 1; y1 < 8 && map[x][y1]; y1++) { if(map[x][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(x1 = x - 1, y1 = y - 1; x1 >= 0 && y1 >= 0 && map[x1][y1]; x1--, y1--) { if(map[x1][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(x1 = x + 1, y1 = y + 1; x1 < 8 && y1 < 8 && map[x1][y1]; x1++, y1++) { if(map[x1][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(x1 = x + 1, y1 = y - 1; x1 < 8 && y1 >= 0 && map[x1][y1]; x1++, y1--) { if(map[x1][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } sign = false; for(x1 = x - 1, y1 = y + 1; x1 >= 0 && y1 < 8 && map[x1][y1]; x1--, y1++) { if(map[x1][y1] == b) sign = true; else { if(sign) sign1 = true; break; } } return sign1; } bool win(void) //判断是否有棋可走 { int x, y; for(x = 0; x < 8; x++) for(y = 0; y < 8; y++) if(judge(x, y, today)) return true; return false; } bool quit(void) //判断是否有棋 { int x, y; bool b = false, w = false; for(x = 0; x < 8; x++) for(y = 0; y < 8; y++) { if(map[x][y] == 'B') b = true; else if(map[x][y] == 'W') w = true; } return(b && w); } bool ask(void) //弹出对话框 { HWND wnd = GetHWnd(); int key; char str[30] = "黑:", s[2]; sprintf(s, "%d", black); strcat(str, s); strcat(str, " 白:"); sprintf(s, "%d", white); strcat(str, s); strcat(str, " 是否重新开始?"); if(black == white) key = MessageBox(wnd, str, "和局", MB_YESNO | MB_ICONQUESTION); else if(black > white) key = MessageBox(wnd, str, "黑胜", MB_YESNO | MB_ICONQUESTION); else key = MessageBox(wnd, str, "白胜", MB_YESNO | MB_ICONQUESTION); if(key == IDYES) return true; else return false; } void play(void) //游戏过程 { MOUSEMSG m; int x, y; for(x = 0; x < 8; x++) for(y = 0; y < 8; y++) map[x][y] = 0; map[3][4] = map[4][3] = 'B'; map[3][3] = map[4][4] = 'W'; today = 'B'; print(); print1(); do { do { while(true) { m = GetMouseMsg(); //获取鼠标消息 if(m.uMsg == WM_LBUTTONDOWN) //如果左键点击 break; } x = m.y / 60; y = m.x / 60; if(judge(x, y, today)) //如果当前位置有效 { draw(x, y, today); //下子 today = ((today == 'B') ? 'W' : 'B'); print1(); //交换 } }while(win() && quit()); //如果当前无棋可走 today = ((today == 'B') ? 'W' : 'B'); print1(); }while(win() && quit()); //双层判断 } int main(void) //主函数 { initgraph(640, 480); load(); setbkcolor(WHITE); cleardevice(); do { play(); }while(ask()); closegraph(); return 0; }
大家赶紧去动手试试吧!
审核编辑:汤梓红
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
游戏
+关注
关注
2文章
794浏览量
27553 -
C语言
+关注
关注
183文章
7646浏览量
146114 -
编程
+关注
关注
90文章
3723浏览量
97430 -
源码
+关注
关注
8文章
689浏览量
31460
原文标题:C语言零基础项目:黑白棋游戏!详细思路+源码分享
文章出处:【微信号:cyuyanxuexi,微信公众号:C语言编程学习基地】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
热点推荐
llabview黑白棋小游戏制作详细解析
`相关课程推荐:《8周LabVIEW视觉项目编程实战特训营》上面这张是效果图编写这样一个界面可以使用到多种不同的思路和技巧,我会按照从简到繁的顺序,分几次来介绍几个不同的方法。 讲由NI 软件工程师阮奇桢为您讲解。llabview黑白棋
发表于 11-23 15:44
【每周一练】LabVIEW益智游戏 - 黑白棋挑战赛&前篇
: ①实现8*8的方格,能够实现黑白棋的基本逻辑; ②状态显示,显示当前是黑手下棋还是白手下棋; ③使用文件菜单,提供配置IP和端口的功能; ④点击新游戏,从新开始; ⑤不需要人机对战,手动走棋,便于第二期
发表于 11-22 19:27
labview制作黑白棋
想用labview制作黑白棋(用于学业交作业),现在做到图片这个地方,不知道下一步该怎么做了,希望有会的烧友教一下,可以加我QQ:871596228或者微信:dmcyljjzg
发表于 05-24 13:25
添加了黑白棋自动下棋功能
之前用 LabVIEW 编写了一个[黑白棋程序],作为学习 XControl 的示例。那个程序基本完整,但是缺少一个 AI 自动走子的功能。这两天有空,把缺的功能补上去了。下棋的水平并不高,但好于随机走子,正好可以和初学的人下的有来有回。代码连接
发表于 08-16 08:12
labview教程之如何使用labview编写一个黑白棋游戏的界面为例
我们需要一个具体示例来帮助介绍这些的技巧,我打算以编写一个黑白棋游戏的界面为例。选择黑白棋是因为这个游戏的界面在常见棋类中比较简单,适合做范例。另外,它也是我最开始学习LabVIEW时
发表于 09-20 15:20
•42次下载
C语言零基础项目:黑白棋游戏!详细思路+源码分享
评论