《c语言课程设计——扫雷.pdf》由会员分享,可在线阅读,更多相关《c语言课程设计——扫雷.pdf(26页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、1 C语言程序设计报告题目:扫雷游戏班级:网 101组长:成员:,日期:目录2 一基本情况二游戏规则三设计思路四函数说明1.游戏控制部分2.图形处理部分3.鼠标处理部分五调试与测试六附录一基本情况开发平台:win-tc(windows-xp)3 开发进度:小组进行图形处理及鼠标相关资料收集,掌握相关图形处理及鼠标相关技术小组成员讨论程序实现算法及任务分配代码实现,调试任务划分:陈超 负责程序总体设计,游戏控制部分实现,后期调试张艳 负责程序图形处理及鼠标处理部分实现,总结文档编写二游戏规则4 玩家进入游戏界面后,只能通过鼠标进行操作,一旦有键盘输入,则退出游戏程序。游戏框图左上方显示的数字为此
2、局游戏中包含的地雷数目。中间黄色笑脸为重新开始按钮,一旦鼠标左键单击此位置,则退出正在进行的扫雷游戏,游戏重新开始。游戏中包含10*10 个空格,可以通过鼠标左键单击或者右键单击来进行扫雷游戏。鼠标单击表示打开此空格位置,若此位置为地雷,则输出所有的地雷分布,玩家输,游戏结束,等待玩家的选择退出或者继续开始下一次游戏。若此位置已经被左单击过,则再次进行鼠标左键单击时,游戏不做任何响应。单击过后若此位置无雷则显示空格或者数字。空格表示此处无雷,数字表示此处无雷且周围八个格子中的地雷数目为空格中的数字数目,若某个空格周围亦没有地雷分布则继续以每个方格为中心继续判断无雷就展开,显示数字或者空格,直至
3、遇到地雷或者是边界。鼠标右键单击表示游戏玩家标识此位置为地雷,若游戏玩家在某位置右键单击过之后,再次右键单击此位置,则表示取消此处标注的地雷标志,若在此处鼠标左键单击,不做任何响应。游戏玩家依次对每一个空格,当游戏玩家把所有的非雷区域全部处理完毕,则游戏结束,玩家胜利,输出“YOU WIN”,等待玩家进一步选择。三设计思路5 扫雷程序主要用了一个10 行 10 列的二维数组,数组的每个元素都是结构体类型,结构体中的num代表格内当前处于什么状态,值1 表示有雷,值 0表示已经变成空白格或者显示过数字,roundnum统计每个格子周围有多少地雷,flag是鼠标右键的标志,如果鼠标左键单击此方格,
4、则对应的二维数组结构体里的 flag为 1 表示格子显示红旗,这样鼠标左键点击在这个格子上无效。首先通过生成随机数的方式布置地雷,如果在该点生成的随机数为1,则该点存在地雷,将NUM(结构体)标为1,随机数为其它值NUM(结构体)标为 0,每生成一个地雷,则全局变量minenum自增一。在程序一开始统计好每个格子周围有多少地雷,将统计的结果存放在对应的结构体变量roundnum,然后当鼠标左键点击在没有地雷的格子上时进行两种判断,如果格子周围没有地雷,就在原来的格子位置显示空白格,然后用递归的方法同样判断周围的8 个格子;如果格子周围有地雷,就在该空格显示具体的雷数。在递归判断8 个格子时,如
5、果格子上有雷或者格子已经显示过雷数或者空白,以及格子上有红旗标志的话,就不再对格子进行任何判断。否则,当此处无雷,即展开此处,并且以此位置为中心继续进行递归,直至遇到边界或者地雷,跳出递归。当游戏玩家每处理一个非雷区域的格子,则对应的统计非雷方格数目的变量自减一,当该变量的值自减为0 时,则表示游戏结束,该玩家胜利。具体函数流程图如下:6 7 四函数说明游戏控制部分Amain()主函数定义使用到的常数,全局变量,函数原型说明。然后初始化图形系统,调用游戏控制函数,按任意键结束游戏,关闭图形系统,游戏结束。Bvoid SpreadMine(int t)布雷函数通过函数srand(int)time
6、(0);生成随机数,随后在二重循环里,生成随机数,若生成的随机数为1,则将 num(结构体)标为 1,表示改点有雷,并在该函数内统计该次游戏中地雷总数,通过 outtextxy(210,70,randminenum);在(210,70)点输出雷的总数,用一百减去总雷数表示该次无雷空格的总数(在输赢判断中,若所有无雷区域都已经得到了处理,即玩家胜利)。C.void GameOver()游戏结束如果单击了地雷,则地雷爆炸,游戏结束,显示地雷分布,两重循环判断每个格子的状态,如果状态值为1,则为地雷,设置红色背景,实体填充模式,画圆表示地雷。D.void GameWin()玩家胜利如果所有的非雷区域
7、的格子都处理过,就表示游戏玩家胜利,输出you win 信息。GameBegain()游戏开始这个函数完成初始界面的设计以及调用函数void SpreadMine(int t)生成地雷。初始界面的主要工作是确定图的位置和方格显示的位置。外边框的左上角为(190,60),右上角坐标为(390,290),显示笑脸和地雷数的区域为8(190,60)(390,90),每个方格的宽度和高度均为16.Fvoid ShowWhite(int i,int j)显示空白区域当单击某个格子,该格子周围的地雷数为0 时,应该继续判断它周围8个方向的格子的周围的雷数是否为0,将这些格子都用空白显示。具体实现如下:a如
8、果当前格子有红旗或者已经显示过数字或空白格子,则返回。b空白格子数减一(统计处理过多少格子)。c如果周围的地雷数为0,且它不是地雷,则将它显示为空白,同时将它的状态值 num置 0,表示处理过;如果周围地雷数不为0,显示周围地雷数,同时将它的状态值 num置 0,表示处理过,返回。G void Control()调用函数 GameBegain()显示游戏初始化界面,调用GamePlay()函数具体实现游戏操作,该函数的返回值有可能有两种:一是正在玩中,提前结束游戏;一是玩完,要么失败,要么胜利。如果是前者,则重新开始游戏。如果是后者,则判断是否单击了笑脸,是则重新开始,否则结束程序。Hvoid
9、 GamePlay()游戏控制游戏过程主要是对鼠标按键的处理,集体算法实现如下:a.如果单击了鼠标左键则判断如果单击了笑脸,则游戏重新开始如果单击了方格,判断该格子有红旗,则按键无效。如果单击的格子没有显示过数字或空白,则判断:如果是地雷,游戏结束,显示地雷分布;如果不是地雷,则统计周围(8 个方向)的地雷数,如果周围地雷数为0,条用递归函数ShowWhite()处理周围格子的情况,显示周围的格子数或者空白。如果单击的格子周围地雷数不为0,则显示周围地雷数,将处理过的格子做标记。如果处理完所有的格子吗,则游戏胜利,显示胜利信息。b.如果单击鼠标右键,该格子没有红旗则显示红旗。如果有红旗,再击右
10、键,9 则红旗消失。c 重复步骤 b 直到按键结束游戏I int MineStatistics(int i,int j)统计每个格子周围的地雷数,分别考虑各自处于四个角,四条边以及中间某个位置的情况。周围是上,下,左,右,左上,左下,右上,右下。在统计四个角格子周围的地雷数目时,统计其周围的三个方格内是否有地雷,如统计左上角位置周围的地雷数目时,其周围的三个方格的二维坐标依次为(i,j+1),(i+1,j),(i+1,j+1)(设该位置坐标为(i,j)。而统计四条边上某一空格周围的地雷数目时,则统计其周围的五个方格是否有地雷,如统计左边的边上某位置的地雷数目时,其周围的五个方格的二维坐标依次为
11、(i-1,j),(i-1,j+1),(i,j+1),(i+1,j),(i+1,j+1)(设该位置坐标为(i,j)。而统计普通位置时要统计其周围的九个方格内是否有地雷,则该点周围的九个坐标依次为(i-1,j-1),(i-1,j),(i-1,j+1),(i,j-1),(i,j+1),(i+1,j-1),(i+1,j-1),(i+1,j),(i+1,j+1).统计完成后,将统计结果存入对应的结构体的roundnum变量中去。10 2.图形处理部分Avoid Init()对计算机进行图形模式初始化,加载图形驱动包Bvoid DrawSmile()画一个笑脸Cvoid DrawRedFlag(int i
12、,int j)画一个红旗Dvoid DrawEmpty(int i,int j,int mode,int color)画两种空格Evoid Close()关闭图形模式11 3.鼠标处理部分Avoid MouseOn()鼠标显示Bvoid MouseOff()鼠标隐藏Cvoid MouseSetXY(int x,int y)设置鼠标当前位置Dint LeftPress()鼠标左键按下Eint RightPress()鼠标右键按下Fvoid MouseGetXY()得到鼠标当前位置12 五调试与测试1.书写标识符时,忽略了大小写字母的区别编译程序把 i 和 I 认为是两个不同的变量名,而显示出错信
13、息。C语言认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。2.在程序嵌套比较多时,容易出错在程序嵌套比较多时,很容易多卸“”,或者少写“”,造成编译错误,为减少类似错误出现,在写程序时应该采取缩进式写法,让处于同一层次的函数处于同一条线上,这样就能够减少上述错误的出线3.win7 不支持 C语言图形全屏模式,图形处理须加载图形驱动在进行图形部分处理时win7 不支持 c 语言图形的全屏模式,在XP系统下,也没法正确运行。原因是C 语言在进行图形处理必须正确设置图形驱动器和模式,通过函数 detectgraphics能自动进行计算机硬件检测,后再
14、进行初始华即可。4.布雷随机数生成的错误以及优化在进行布雷函数编写时,首先用随机数生成函数使随机数生成范围在(0-1)之间,使得地雷主要分布与上部,下部区域基本没有地雷,为了使得雷区较为平均,我又加了一重循环,强制每行最多只能分布2 个地雷,并另为生成两个随机数,强制是随机数的行数,该行不分布地雷,这样使得雷区相对平均,但语句过于繁琐,且对于知道该布雷规则的人,游戏更加容易,最后我们采用了加大随机数生成范围,让随机数在(0,t)之间,t 可以调节就是得雷区均匀的同时,可以通过调节 t 的值来控制某局地雷数目的多少。而在进行生成随机数时时间种子13 的生成位置会影响随机数的生成,因为在计算机中随
15、机数一旦生成,计算机就不会再生成随机数,会导致生成的随机数相同。5 在调用递归函数及递归函数编写时的错误在调用递归函数前应该先调用MineStatistics函数统计该格子周围是否无雷,只有当该格子无雷且周围无雷时(而不时该格子处无雷即可),才能调用递归函数向八个方向递归展开无雷区域,该递归函数的出口是碰到了已经处理过的格子或者该格子已经做了标记,又或者是该格子有雷,而在进行递归在向周围的八个方向展开时要注意设置边界值的情况,否则会造成处理数据超出数组定义的边界,造成不必要的错误6.MineStatistics函数编写时条件不易设置函数 MineStatistics编写时不同情况进行分别统计时
16、条件设置很容易错误,类似于鞍点查找时的错误,导致一类点周围的雷数的统计出现错误,最后我采用在纸上画出模型,在进行某类点周围雷数统计编写时,根据纸上模型的规律进行编写7.输赢判断是靠检索非雷的格子是否处理完游戏的输赢判断有两种思路,一是检索是雷的格子是否已经全部做处理(即标红旗),二是检索非雷的格子是否已经全部处理完全。雷的数目虽然较少,但是雷点具有不确定性,为了实现方法一的判断,还需定义一个二维数组,用于保存哪些位置是有地雷分布的,在进行判断时,首先读取该数组的标记,若该格子有雷,再从结构体数组中读取flag的值,看该格子的地雷是否已经做了处理。而,进行方法二的实现只需要每处理一个非雷的格子,
17、让统计非雷数目的数据自减一,当其自减为 0 时,则表示玩家取得了胜利,比较两种处理方法,知道方法二比较容易实现8.该函数鼠标的光标没法显示出来14 该函数鼠标的光标没法显示出来,在查询了有关鼠标光标显示后,进行了鼠标光标显示的尝试。但出现了光标只有在鼠标单击后才能显示,通过条件设置使得能在移动时显示鼠标的位置,但鼠标光标的图形会遮盖原有图形的显示。该进方法有两种,一是每次进行鼠标显示后进行清屏处理,但这样需要在清屏前保存所有位置点的信息,数据的保存,读取较复杂,没有实现;二是,在每次进行鼠标光标输出前通过数据记录改点的图形信息,在输出鼠标,且鼠标移出改点后,用原来的图形信息通过图形输出覆盖掉鼠
18、标光标显示留下的痕迹还原原有图形,该方法的实现也较为复杂,亦没有实现。由于该部分功能没有实现,没有将本部分相关的代码附录出来15 六附录#include#include#include#include#define LEFTPRESS 0 xff01#define LEFTCLICK 0 xff10#define LEFTDRAG 0 xff19#define MOUSEMOVE 0 xff08void Init();void Close();void DrawSmile();void DrawRedFlag(int i,int j);void DrawEmpty(int i,int j,in
19、t mode,int color);void Control();int MineStatistics(int i,int j);void SpreadMine(int t);void ShowWhite(int i,int j);void GamePlay();void GameBegain();void GameWin();void GameOver();void MouseOn();void MouseOff();void MouseSetXY(int x,int y);int LeftPress();int RightPress();void MouseGetXY();16 char
20、randminenum3;int minenum;int MouseX,MouseY;int gameAgain=0;int gamePlay=0;struct int num;/*无雷标 0,有雷标 1*/int roundnum;/*此位置有雷标 0,否则统计周围九个方格内的雷数*/int flag;/*标记为雷时为 1,没有标记为 0*/Mine1010;main()Init();Control();Close();getch();void MouseOn()/*鼠标光标显示*/_AX=0 x01;geninterrupt(0 x33);void MouseOff()/*鼠标光标隐藏*/
21、_AX=0 x02;geninterrupt(0 x33);void MouseSetXY(int x,int y)/*设置鼠标当前位置*/_CX=x;_DX=y;_AX=0 x04;geninterrupt(0 x33);int LeftPress()/*鼠标左键按下*/17 _AX=0 x03;geninterrupt(0 x33);return(_BX&1);int RightPress()/*鼠标右键按下*/_AX=0 x03;geninterrupt(0 x33);return(_BX&2);void MouseGetXY()/*得到当前位置*/_AX=0 x03;geninterr
22、upt(0 x33);MouseX=_CX;MouseY=_DX;void Init()/*图形初始化*/int gdriver,gmode;registerbgidriver(EGAVGA_driver);/*加载图形驱动包*/detectgraph(&gdriver,&gmode);/*检测硬件确定图形驱动程序和模式*/initgraph(&gdriver,&gmode,);/*初始化图形系统,gdriver和gmode分别表示图形驱动器和模式,内为图形驱动的路径*/void DrawSmile()/*花笑脸*/setfillstyle(SOLID_FILL,YELLOW);/*设置填充模
23、式和颜色,表式对下面所画图形的填充,此处表示对下面椭圆的填充*/fillellipse(290,75,10,10);/*画椭圆,前两个参数表示中心点坐标,后面依次为一个宽度和一个高度*/setcolor(YELLOW);/*设置画笔颜色*/setfillstyle(SOLID_FILL,BLACK);/*此处表示对下面两个椭圆的填充*/fillellipse(285,75,2,2);fillellipse(295,75,2,2);setcolor(BLACK);bar(287,80,293,81);/*画一个二维线条*/18 void DrawRedFlag(int i,int j)setco
24、lor(7);/*设置画笔颜色*/setfillstyle(SOLID_FILL,RED);bar(198+j*20,95+i*20,198+j*20+5,95+i*20+5);setcolor(BLACK);line(198+j*20,95+20*i,198+j*20,95+i*20+5);void DrawEmpty(int i,int j,int mode,int color)setcolor(color);setfillstyle(SOLID_FILL,color);if(mode=0)/*没有单击过显示大格子*/bar(200+j*20-8,100+i*20-8,200+j*20+8
25、,100+i*20+8);if(mode=1)/*单击过后显示小格子*/bar(200+j*20-7,100+i*20-7,200+j*20+7,100+i*20+7);void Close()closegraph();void SpreadMine(int t)int i,j;minenum=0;srand(int)time(0);/*生成时间种子*/for(i=0;i10;i+)for(j=0;j10;j+)Mineij.num=rand()%t;if(Mineij.num=1)minenum+;else Mineij.num=2;Mineij.flag=0;sprintf(randmin
26、enum,%d,minenum);setcolor(1);settextstyle(0,0,2);19 outtextxy(210,70,randminenum);minenum=100-minenum;/*变量取空白格数量*/MouseOn();void GameOver()int i,j;setcolor(0);for(i=0;i10;i+)for(j=0;j10;j+)if(Mineij.num=1)DrawEmpty(i,j,0,RED);setfillstyle(SOLID_FILL,BLACK);fillellipse(200+j*20,100+i*20,7,7);void Gam
27、eWin()setcolor(1);settextstyle(0,0,2);outtextxy(230,30,You Win);int MineStatistics(int i,int j)int nNUM=0;if(i=0&j=0)/*统计左上角*/if(Mine01.num=1)nNUM+;if(Mine10.num=1)nNUM+;if(Mine11.num=1)nNUM+;else if(i=0&j=9)/*统计右上角*/if(Mine08.num=1)nNUM+;if(Mine19.num=1)nNUM+;20 if(Mine18.num=1)nNUM+;else if(i=9&j=
28、0)/*统计左下角*/if(Mine91.num=1)nNUM+;if(Mine80.num=1)nNUM+;if(Mine81.num=1)nNUM+;else if(i=9&j=9)/*统计右下角*/if(Mine98.num=1)nNUM+;if(Mine89.num=1)nNUM+;if(Mine88.num=1)nNUM+;else if(i=0&j!=0&j!=9)/*统计第一行*/if(Mineij+1.num=1)nNUM+;if(Mineij-1.num=1)nNUM+;if(Minei+1j-1.num=1)nNUM+;if(Minei+1j.num=1)nNUM+;if(
29、Minei+1j+1.num=1)nNUM+;else if(i=9&j!=0&j!=9)/*统计最后一行*/if(Mineij+1.num=1)nNUM+;if(Mineij-1.num=1)nNUM+;if(Minei-1j-1.num=1)nNUM+;if(Minei-1j.num=1)nNUM+;21 if(Minei-1j+1.num=1)nNUM+;else if(j=0&i!=0&i!=9)/*统计第一列*/if(Minei-1j.num=1)nNUM+;if(Minei+1j.num=1)nNUM+;if(Mineij+1.num=1)nNUM+;if(Minei-1j+1.n
30、um=1)nNUM+;if(Minei+1j+1.num=1)nNUM+;else if(j=9&i!=0&i!=9)/*统计最后一列*/if(Minei-1j.num=1)nNUM+;if(Minei+1j.num=1)nNUM+;if(Mineij-1.num=1)nNUM+;if(Minei-1j-1.num=1)nNUM+;if(Minei+1j-1.num=1)nNUM+;else if(Minei-1j-1.num=1)nNUM+;if(Minei-1j.num=1)nNUM+;if(Minei-1j+1.num=1)nNUM+;if(Mineij-1.num=1)nNUM+;if
31、(Mineij+1.num=1)nNUM+;if(Minei+1j-1.num=1)nNUM+;if(Minei+1j.num=1)22 nNUM+;if(Minei+1j+1.num=1)nNUM+;return(nNUM);void GameBegain()int i,j;cleardevice();if(gamePlay!=1)MouseSetXY(290,70);/*鼠标开始位置,并作为初时坐标*/MouseX=290;MouseY=70;gamePlay=1;/*下次开始鼠标不重新初始化*/minenum=0;setfillstyle(SOLID_FILL,7);bar(190,60
32、,390,290);for(i=0;i10;i+)for(j=0;j280&MouseX65&MouseY85)gameFlag=1;continue;if(kbhit()break;MouseOff();void GamePlay()int i,j,num;/*num用于接收一个格子周围右多少地雷*/for(i=0;i10;i+)for(j=0;j280&MouseX65&MouseY190&MouseX90&MouseY290)/*当前鼠标处于格子范围内*/j=(MouseX-190)/20;/*x坐标*/i=(MouseY-90)/20;/*y坐标*/if(Mineij.flag=1)/
33、*如果格子有红旗,则按键无效*/continue;25 if(Mineij.num!=0)/*如果格子没有处理过*/if(Mineij.num=1)/*如果按下的格子是雷*/MouseOff();GameOver();/*游戏失败*/break;else /*鼠标按下的格子不是雷*/MouseOff();num=MineStatistics(i,j);if(num=0)/*周围没有雷就用ShowWhite显示空白格子*/ShowWhite(i,j);else /*周围有雷*/sprintf(randminenum,%d,num);/*输出当前格子周围的雷数*/setcolor(RED);out
34、textxy(195+j*20,95+i*20,randminenum);minenum-;MouseOn();Mineij.num=0;/*点过的格子周围num 变成 0,表示这个格子已经处理过*/if(minenum190&MouseX90&MouseY290)/*当前鼠标处于格子范围内*/j=(MouseX-190)/20;/*x坐标*/i=(MouseY-90)/20;/*y坐标*/26 MouseOff();if(Mineij.num!=0&Mineij.flag=0)DrawRedFlag(i,j);Mineij.flag=1;else if(Mineij.flag=1)DrawEmpty(i,j,0,8);Mineij.flag=0;MouseOn();sleep(1);