手把手教你用vc6做五子棋小游戏_第1页
手把手教你用vc6做五子棋小游戏_第2页
手把手教你用vc6做五子棋小游戏_第3页
手把手教你用vc6做五子棋小游戏_第4页
手把手教你用vc6做五子棋小游戏_第5页
已阅读5页,还剩27页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、第三章 五子棋五子棋是一种很受人们喜爱的游戏,它的规则简单,但玩法变化多端,富有趣味性,适合人们消遣。这里我们就来设计一个五子棋游戏。(一) (一) 人对人游戏 1. 1. 游戏实现人对人游戏,其实只是对游戏规则的实现,我们只是利用五子棋游戏的规则来编程,至于真正的游戏实现计算机的“智能”算法,我们将在后面讲述。五子棋的规则很简单:1,判断是否能放下棋子(是否已经有了棋子);2,判断是哪种颜色下棋;3,判断是否已经结束(是谁赢?)。这些规则,我们将用相应的函数来实现。其它,我们还将介绍其它一些功能的实现。如鼠标的更换,工具栏和状态栏的编辑,类与类之间的相互调用。新建工程3_1,选择单文档,在s

2、tep 4 of 6中先中windows sockets复选框。如下图: 图3-1-12. 2. 资源编辑由于我们这个程序出现的关于资源编辑的内容太多,我们具体介绍如下:见下图3-1-2,我们需要添加的有: 图3-1-2 黑白位图bitmap以表示棋盘上面的棋子:idb_black idb_white黑白鼠标cursor以替换当前鼠标:idc_cursor1黑棋子idc_cursor2白棋子说明:由于下棋时我们必须把鼠标热点设置在中间,点击下图(图3-1-3)最右边按扭,然后把鼠标移动到图像中你想设置为热点的地方,按下鼠标左键。 图3-1-3黑白图标icon以显示在状态栏供以提示:idi_bl

3、ackidi_white 说明: 由于我们的图标支持256色,按下下图(图3-1-4)最右边按扭,选择device里面显示的选项。 图3-1-4菜单以供操作:开始:id_start保存:id_save打开:id_open工具栏:如上图所示。说明:工具栏一般都是根据菜单选项而产生的,它的id一般都能从菜单的id 中找到。3. 3. 变量函数首先,为了实现状态栏的应用,我们必须更改它的变量:在mainfrm.h文件里面,把cstatusbar m_wndstatusbar 为public接着是在3_1view.h文件里面添加变量函数: /两个鼠标 hcursor hcursorwhite;hcur

4、sor hcursorblack; /棋盘数组int wzq1919; / colorwhite true时白棋下,否则黑棋下bool colorwhite; /棋子位图cbitmap m_bmblack;cbitmap m_bmwhite;/保存文件void save();/检查是否结束void over(cpoint point);/鼠标操作afx_msg void onlbuttonup(uint nflags, cpoint point);/鼠标图形更换afx_msg bool onsetcursor(cwnd* pwnd, uint nhittest, uint message);/

5、菜单的开始afx_msg void onstart();/菜单的保存afx_msg void onsave();/菜单的打开afx_msg void onopen();4. 4. 具体实现棋盘大小设置:由于我们的游戏的棋盘大小是一定的,不能改变大小的,是应该符合要求的。在如下函数添加设置窗口大小的语句:bool cmainframe:precreatewindow(createstruct& cs)if( !cframewnd:precreatewindow(cs) )return false;/ todo: modify the window class or styles here by

6、modifying/ the createstruct cscs.dwexstyle=cs.dwexstyle|ws_ex_topmost;/cs.style=ws_sysmenu|ws_overlapped|ws_minimizebox;/; /设置窗口大小:400*340cs.cx=450;cs.cy=500; return true;初始化变量:在构造函数里添加初始代码:cmy3_1view:cmy3_1view()/ todo: add construction code here/load鼠标图像和棋子位图hcursorblack=afxgetapp()-loadcursor(idc

7、_cursor1);hcursorwhite=afxgetapp()-loadcursor(idc_cursor2);m_bmwhite.loadbitmap(idb_white);m_bmblack.loadbitmap(idb_black);/清理棋盘 /数组值为0表示没有棋子for(int i=0;i19;i+)for(int j=0;jfillrect(myrect1,&mybrush1);/画棋盘框线cpen mypen;cpen*myoldpen;mypen.createpen(ps_solid,1,rgb(0,0,0); myoldpen=pdc-selectobject(&my

8、pen); for(int i=0;imoveto(40,40+i*20);pdc-lineto(400,40+i*20);pdc-moveto(40+i*20,40);pdc-lineto(40+i*20,400); /重画时显示存在的棋子cdc dc;if(dc.createcompatibledc(pdc)=false) afxmessagebox(cant create dc);for(int n=0;n19;n+)for(int m=0;mbitblt(n*20+32,m*20+32,160,160,&dc,0,0,srccopy);else if(wzqnm=-1)/显示黑棋dc.

9、selectobject(m_bmblack);pdc-bitblt(n*20+32,m*20+32,160,160,&dc,0,0,srccopy);设置鼠标:棋盘画好了,接下来就是下棋了。但鼠标并没有像我们上面说的那样变成白棋,加函数如下:bool cmy3_1view:onsetcursor(cwnd* pwnd, uint nhittest, uint message) / todo: add your message handler code here and/or call defaultif(nhittest=htclient)/白棋下,显示白棋鼠标if(colorwhite)

10、/调用主框架里面的状态栏cmainframe*pfrm=(cmainframe*)afxgetapp()-m_pmainwnd;cstatusbar*pstatus=&pfrm-m_wndstatusbar;if(pstatus)pstatus-getstatusbarctrl().seticon(0,afxgetapp()-loadicon(idi_white);pstatus-setpanetext(0,白棋下);setcursor(hcursorwhite);/显示黑棋鼠标elsesetcursor(hcursorblack);cmainframe*pfrm=(cmainframe*)a

11、fxgetapp()-m_pmainwnd;cstatusbar*pstatus=&pfrm-m_wndstatusbar;if(pstatus)/显示图像pstatus-getstatusbarctrl().seticon(0,afxgetapp()-loadicon(idi_black);/显示文字pstatus-setpanetext(0,黑棋下);return 1;return cview:onsetcursor(pwnd, nhittest, message);现在运行程序,怎样,鼠标变成白棋了,而且下面的状态栏也能够显示鼠标状态了,真是一举两得。可是,又该怎样把棋子放在棋盘上呢?下

12、棋操作:这就涉及到onlbuttondown(uint nflags, cpoint point)和onlbuttonup(uint nflags, cpoint point)两个函数了。要用哪一个或用两个?用down函数时是在鼠标按下时放下棋子,可是,要是我们按下后意识到按错了怎么办;那就改用up函数,表示当鼠标键松开时放下棋子。ok!添加函数如下:void cmy3_1view:onlbuttonup(uint nflags, cpoint point) / todo: add your message handler code here and/or call defaultcdc *p

13、dc=getdc();cdc dc;if(dc.createcompatibledc(pdc)=false) afxmessagebox(cant create dc); /是否在棋盘内 if(point.x30&point.x30&point.ybitblt(px*20+32,py*20+32,160,160,&dc,0,0,srccopy);/表示存在白棋wzqpxpy=1;/检查是否结束over(point);/换黑棋下colorwhite=false;else if(wzqpxpy=0)dc.selectobject(m_bmblack); pdc-bitblt(px*20+32,py

14、*20+32,160,160,&dc,0,0,srccopy);wzqpxpy=-1;over(point);colorwhite=true;cview:onlbuttonup(nflags, point);由上面可以看出,当鼠标键松开时判断,如果那个位置没有棋子,则放下,并把棋盘数组赋相应的值:1 或-1。是否结束:接着是用一个over()函数判断是否结束,是则结束并重新开始;否则,接着把鼠标变成对方棋子,表示对方下棋。那over()函数又是怎样的呢? 此函数是利用刚下棋的位置为中心,检查它各个方向上的连续五个棋子是否同色,是则结束并重新开始。然而,我们又是怎样判断一个方向上的五个棋子的同色

15、的?这就涉及地为什么我要把五子棋数组赋值为1和-1的问题。因为这样有一个好处:利用连续五个棋子的值相加,如果它们的值的绝对值等于5,则说明是同色。当然,这只是这样赋值的一点作用,真正的作用将在后面介绍。添加如下:void cmy3_1view:over(cpoint point)/获取鼠标指向数组位置,即中心位置int x=(point.x-30)/20;int y=(point.y-30)/20;/计算开始判断的坐标 xx,yyint xx,yy; if(x4)xx=0;elsexx=x-4;if(y4)yy=0;elseyy=y-4;int i,j,a;/横向判断for(i=xx;i15;

16、i+) a=0;for(j=i;ji+5;j+)a=a+wzqjy;/五个都是白棋 if(a=5)afxmessagebox(白棋胜!);/重新开始onstart();return;/五个都是黑棋if(a=-5)afxmessagebox(黑棋胜!);onstart();return;/竖向判断for(i=yy;i15;i+) a=0;for(j=i;ji+5;j+)a=a+wzqxj; if(a=5)afxmessagebox(白棋胜!);onstart();return;if(a=-5)afxmessagebox(黑棋胜!);onstart();return;/向右下角/判断起点位置if(

17、xy)if(xx=0)yy=y-x;elseif(yy=0)xx=x-y;/参数over=1时退出循环int over=0;doa=0;for(i=0;i5;i+)if(xx+i)19|(yy+i)(18-x)if(x13)yy=y-(18-x);xx=18;elseyy=y-4;xx=x+4;elseif(y5)xx=x+y;yy=0;elseyy=y-4;xx=x+4; over=0;doa=0;for(i=0;i=0|(yy+i)19)a=a+wzqxx-iyy+i;if(a=5)afxmessagebox(白棋胜!);onstart();return;if(a=-5)afxmessag

18、ebox(黑棋胜!);onstart();return;/到了边界elseover=1; xx-=1;yy+=1;while(over=0); 现在,我们的人对人游戏就完成了。下面介绍附加内容。5. 5. 附加内容理论上,这个游戏并无须保存,因为它确实太小了。事实上,这个游戏有保存的功能,由于我们学习的需要。这个游戏的保存,与其说是学习文件的保存,不如说是我们学习字符串的操作。另外,这个附加的内容并不是为了当前的学习而添加的,而是为了后面的学习和应用而铺垫的。 保存文件:保存文件函数是一个菜单选项。它的作用就是保存当前游戏的状态。首先,我们应该为我们自己的文件定义一个后缀名:.wzq;接着是打

19、开保存文件的公共对话框,如果确定,则表示保存,那么就先获取文件名,然后按照一定的顺序保存各个点的数组的值,最后保存当前是哪种颜色下棋。void cmy3_1view:onsave() / todo: add your command handler code here/设置保存的文件,后缀名wzqcfiledialog dlg(false,wzq,null,ofn_hidereadonly|ofn_overwriteprompt,(*.wzq)|*.wzq|all files|*.*|,this);/如果公共类对话框为确定if(dlg.domodal()=idok) /获取文件名dlg.get

20、filename();/否则,退出elsereturn;/字符串变量cstring str;int i,j; cstdiofile file;/如果有问题,退出if(file.open(dlg.getfilename(),cfile:modecreate|cfile:modewrite|cfile:typetext)=0)afxmessagebox(save error!);return;/循环把棋盘数组的值写进文件for(i=0;i19;i+)for(j=0;j19;j+)if(wzqij=-1)file.writestring(-1n);if(wzqij=0)file.writestrin

21、g(0n);if(wzqij=1)file.writestring(1n);/保存当前下棋颜色if(colorwhite=true)file.writestring(1n);elsefile.writestring(0n);/关闭文件 file.close();读取文件:读文件就是把我们以前保存的文件打开,读取当前打开文件的内容,并给数组赋值使和文件内容相同,然后可以继续进行游戏。/与保存文件相反void cmy3_1view:onopen() / todo: add your command handler code herecfiledialog dlg(true,wzq,null,ofn

22、_hidereadonly|ofn_overwriteprompt,(*.wzq)|*.wzq|all files|*.*|,this);if(dlg.domodal()=idok) dlg.getfilename();elsereturn;cstring str;int i,j,m;cstdiofile file;if(file.open(dlg.getfilename(),cfile:moderead)=0)afxmessagebox(save error!);return;carchive ar(&file,carchive:load);for(i=0;i19;i+)for(j=0;j3

23、0&point.x30&point.ybitblt(px*20+32,py*20+32,160,160,&dc,0,0,srccopy);wzqpxpy=1;over(point);colorwhite=false;/保存白棋位置vspoint=point;/计算机下棋computerdown(); /人对人if(vscomputer=2)if(point.x30&point.x30&point.ybitblt(px*20+32,py*20+32,160,160,&dc,0,0,srccopy);wzqpxpy=1;over(point);colorwhite=false;else if(wz

24、qpxpy=0)dc.selectobject(m_bmblack); pdc-bitblt(px*20+32,py*20+32,160,160,&dc,0,0,srccopy);wzqpxpy=-1;over(point);colorwhite=true;cview:onlbuttonup(nflags, point);由上面可知,我们对人对机游戏的方法是采用:人下完了之后,检查是否胜利,是则结束游戏,重新开始,并改为黑棋(即计算机)先下;如果人没有胜利,也改为计算机下。而对于计算机怎么下棋,我们只是用了一个函数computerdown()表示。但是现在我们并不能运行程序,因为有一个没有定义

25、的函数,怎么办呢?l l 程序扩展思想:添加一个空函数!仔细看程序的话,你还会发现一个变量cpoint vspoint,这是后来添加的它的用处是保存刚才白棋下的位置,有利于黑棋下棋时的定位。接着,我们的主要问题就是实现computerdown()函数,让计算机能够自动下棋!计算机下棋:计算机是怎样下棋?这就是定位的问题了。即搜索棋盘,找出一个最佳点,放下黑棋。我们实现的方法是:全盘搜索,并把搜索到的位置,保存在变量。由于有多种情况,我们定义变量如下:cpoint bpointcan4,/这个位置空,它旁边有四个黑棋wpointcan4,/这个位置空,它旁边有四个白棋bpointcan3,/这个

26、位置空,它的旁边有三个黑棋wpointcan3,/这个位置空,它的旁边有三个白棋bpointcan2,/这个位置空,它的旁边有两个黑棋wpointcan2,/这个位置空,它的旁边有两个白棋bpointcan1;/不是以上情况,这个位置空并在搜索之前都赋值为(-1,-1),然后,进行搜索,并把相应的值保存在相应变量里面,而如果前面已经对变量赋值,我们依然赋值,用新值代替旧值。注意:我们只保存最后一个值,这样的一个好处是,避免了每次都从左上角开始,并且它的随机性比随机函数还随机。全盘搜索完之后,由于上面的变量中至少有一个已经被赋值,即不是(-1,-1),我们可以采用多数优先的方法,让已经有多个同色

27、棋子的位置先下棋。其原理是,如果已经有四个黑棋,计算机再下一个黑棋就赢了;否则,如果人已经有四个白棋,那么计算机就必须放下一个黑棋,阻止白棋下一步赢;如果已经有三个黑棋,再下一个黑棋,变成四个;否则,如果已经有三个白棋,下一个黑棋,破坏它;两个棋子的同理;否则,在刚才白棋下的地方,顺便找一个位置,下棋。computerdown()函数如下:/轮到计算机下棋void cmy3_1view:computerdown()/把各种情形赋值为如下bpointcan4=(-1,-1);wpointcan4=(-1,-1);bpointcan3=(-1,-1);wpointcan3=(-1,-1);bpoi

28、ntcan2=(-1,-1);wpointcan2=(-1,-1);bpointcan1=(-1,-1); /搜索最好的落棋点for(int i=0;i19;i+)for(int j=0;jbitblt(point.x*20+32,point.y*20+32,160,160,&dc,0,0,srccopy); wzqpoint.xpoint.y=-1;/由于原来我们检查是否结束时用的是鼠标点下的坐标,而现在/putdown(cpoint point)函数用的是数组棋盘的坐标,所以必须转换cpoint overpoint;overpoint.x=point.x*20+30;overpoint.y

29、=point.y*20+30;over(overpoint);colorwhite=true; 搜索最佳落棋点:现在就剩下void bestputdown(int i,int j)函数没有定义了(虽然前面的变量函数已经说明了,当时我们这里是用程序扩展的思路进行的,故如此说明)。它的实现原理是:在四个方向上,各自计算那个方向上棋子的状态,我们的思路是利用原来定义的白棋为1,黑棋为-1,的思想,让同个方向上的五个棋子的值相加,取绝对值并赋值给为这个方向定义的局部变量numi。 为什么要用五个棋子的值相加呢?因为,如果几个棋子是同色的,无论黑白,它的绝对值必然大,而对于几个棋子中有黑棋和白棋的,其值必然相

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论