c语言课程设计扫雷_第1页
c语言课程设计扫雷_第2页
c语言课程设计扫雷_第3页
c语言课程设计扫雷_第4页
c语言课程设计扫雷_第5页
已阅读5页,还剩18页未读 继续免费阅读

下载本文档

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

文档简介

1、C语言程序设计报告题目:扫雷游戏班级:网101组长:成员:,日期:2011.9-2011.11一基本情况目录二游戏规则三设计思路四函数说明:1. 游戏控制部分2. 图形处理部分3. 鼠标处理部分P.3p.4p.5p.7p.10p.11p.12五调试与测试六附录p.15一基本情况开发平台: wi n-tc( win dows-xp)开发进度:9.109.20 小组进行图形处理及鼠标相关资料收集,掌握相关图形处理及 鼠标相关技术9.2110.10 小组成员讨论程序实现算法及任务分配10.1011.3代码实现,调试任务划分:陈超 负责程序总体设计,游戏控制部分实现,后期调试张艳负责程序图形处理及鼠标

2、处理部分实现,总结文档编写二游戏规则玩家进入游戏界面后,只能通过鼠标进行操作,一旦有键盘输入,则退出游戏程序。游 戏框图左上方显示的数字为此局游戏中包含的地雷数目。中间黄色笑脸为重新开始按 钮,一旦鼠标左键单击此位置,则退出正在进行的扫雷游戏,游戏重新开始。游戏中包 含10*10个空格,可以通过鼠标左键单击或者右键单击来进行扫雷游戏。鼠标单击表示打开此空格位置,若此位置为地雷,则输出所有的地雷分布,玩家输,游戏结束,等待 玩家的选择退出或者继续开始下一次游戏。 若此位置已经被左单击过,则再次进行鼠标 左键单击时,游戏不做任何响应。单击过后若此位置无雷则显示空格或者数字。空格表 示此处无雷,数字

3、表示此处无雷且周围八个格子中的地雷数目为空格中的数字数目,若某个空格周围亦没有地雷分布则继续以每个方格为中心继续判断无雷就展开,显示数字或者空格,直至遇到地雷或者是边界。鼠标右键单击表示游戏玩家标识此位置为地雷, 若游戏玩家在某位置右键单击过之后, 再次右键单击此位置,则表示取消此处标注的地 雷标志,若在此处鼠标左键单击,不做任何响应。游戏玩家依次对每一个空格,当游戏 玩家把所有的非雷区域全部处理完毕,则游戏结束,玩家胜利,输出“YOU WIN,等待玩家进一步选择。三设计思路扫雷程序主要用了一个10行10列的二维数组,数组的每个元素都是结构体类型,结 构体中的num代表格内当前处于什么状态,值

4、1表示有雷,值0表示已经变成空白格或者 显示过数字,roundnum统计每个格子周围有多少地雷,flag是鼠标右键的标志,如果鼠标 左键单击此方格,则对应的二维数组结构体里的flag为1表示格子显示红旗,这样鼠标左 键点击在这个格子上无效。首先通过生成随机数的方式布置地雷,如果在该点生成的随机数为1,则该点存在地雷,将NUM(结构体)标为1,随机数为其它值NUM(结构体)标为0,每生成一个地雷, 则全局变量minenum自增一。在程序一开始统计好每个格子周围有多少地雷,将统计的结果存放在对应的结构体变 量roundnum,然后当鼠标左键点击在没有地雷的格子上时进行两种判断,如果格子周围没 有地

5、雷,就在原来的格子位置显示空白格,然后用递归的方法同样判断周围的8个格子; vfVa.如果格子周围有地雷,就在该空格显示具体的雷数。在递归判断8个格子时,如果格子上有雷或者格子已经显示过雷数或者空白,以及格 子上有红旗标志的话,就不再对格子进行任何判断。否则,当此处无雷,即展开此处,并且以此位置为中心继续进行递归,直至遇到边界或者地雷,跳出递归。当游戏玩家每处理一个非雷区域的格子,则对应的统计非雷方格数目的变量自减一, 当该变量的值自减为0时,则表示游戏结束,该玩家胜利。具体函数流程图如下: 四函数说明1游戏控制部分A. main ()主函数定义使用到的常数,全局变量,函数原型说明。然后初始化

6、图形系统,调用游戏控制 函数,按任意键结束游戏,关闭图形系统,游戏结束。B. void SpreadMine(int t)布雷函数通过函数srand(int)time(O);生成随机数,随后在二重循环里,生成随机数,若生 成的随机数为1,则将num (结构体)标为1,表示改点有雷,并在该函数内统计该次游戏 中地雷总数,通过outtextxy(210,70,randminenum);在(210,70)点输出雷的总数,用一百减 去总雷数表示该次无雷空格的总数(在输赢判断中,若所有无雷区域都已经得到了处理, 即玩家胜利)。C. void GameOver游 戏结束如果单击了地雷,则地雷爆炸,游戏结束

7、,显示地雷分布,两重循环判断每个 格子的状态,如果状态值为1,则为地雷,设置红色背景,实体填充模式,画圆表示地雷。D. void GameWi n(玩家胜利如果所有的非雷区域的格子都处理过,就表示游戏玩家胜利,输出you win信息。E. void GameBegai n游 戏开始这个函数完成初始界面的设计以及调用函数 void SpreadMi ne(int t)生成地雷。初始 界面的主要工作是确定图的位置和方格显示的位置。外边框的左上角为(190, 60),右上角坐标为(390, 290),显示笑脸和地雷数的区域为(190, 60)(390, 90),每个方格的 宽度和高度均为16.F.

8、void ShowWhite(int i,int j)显示空白区域当单击某个格子,该格子周围的地雷数为0时,应该继续判断它周围8个方向的格子的周围的雷数是否为0,将这些格子都用空白显示。具体实现如下:a如果当前格子有红旗或者已经显示过数字或空白格子,则返回。b空白格子数减一(统计处理过多少格子)。c 如果周围的地雷数为0,且它不是地雷,则将它显示为空白,同时将它的状态值 num 置0,表示处理过;如果周围地雷数不为0,显示周围地雷数,同时将它的状态值num置0, 表示处理过,返回。G. void Control()调用函数GameBegain()显示游戏初始化界面,调用GamePlay()函数

9、具体实现游戏操作,该函数的返回值有可能有两种:一是正在玩中,提前结束游戏;一是玩完,要么 失败,要么胜利。如果是前者,则重新开始游戏。如果是后者,则判断是否单击了笑脸, 是则重新开始,否则结束程序。H. void GamePlay(游戏控制游戏过程主要是对鼠标按键的处理,集体算法实现如下:a. 如果单击了鼠标左键则判断如果单击了笑脸,则游戏重新开始如果单击了方格,判断该格子有红旗,则按键无效。如果单击的格子没有显示过数字或空白,则判断:如果是地雷,游戏结束,显示地雷分布;如果不是地雷,则统计周围(8个方向)的地雷数,如果周围地雷数为 0,条用递归函数ShowWhite ()处理周围格子的情 况

10、,显示周围的格子数或者空白。如果单击的格子周围地雷数不为0,则显示周围地雷数,将处理过的格子做标记。如果处理完所有的格子吗,则游戏胜利,显示胜利信息。b. 如果单击鼠标右键,该格子没有红旗则显示红旗。如果有红旗,再击右键,则红旗消失。c重复步骤b直到按键结束游戏I . int MineStatistics(int i,int j)统计每个格子周围的地雷数,分别考虑各自处于四个角,四条边以及中间某个 位置的情况。周围是上,下,左,右,左上,左下,右上,右下。在统计四个角格子周围 的地雷数目时,统计其周围的三个方格内是否有地雷,如统计左上角位置周围的地雷数目 时,其周围的三个方格的二维坐标依次为(

11、i,j+1 ),(i+1,j ),(i+1,j+1 )(设该位置坐 标为(i , j)。而统计四条边上某一空格周围的地雷数目时,则统计其周围的五个方格是 否有地雷,如统计左边的边上某位置的地雷数目时,其周围的五个方格的二维坐标依次为(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+

12、1,j+1 |).统计完成后,将统计结果存入对应的结构体的roundnum变量中去。2. 图形处理部分A. void Ini t()对计算机进行图形模式初始化,加载图形驱动包B. void DrawSmile()画一个笑脸C. void DrawRedFlag(int i,int j)画一个红旗D. void DrawEmpty(int i,int j,int mode,int color)画两种空格E. void Close()关闭图形模式3.鼠标处理部分 4A. void Mouse0n()鼠标显示mHm B. void MouseOff()鼠标隐藏C. void MouseSetXY(i

13、nt x,int y)D. int LeftPress()鼠标左键按下E. int RightPress() 鼠标右键按下F. void MouseGetXY()得到鼠标当前位置五调试与测试1.书写标识符时,忽略了大小写字母的区别编译程序把i和I认为是两个不同的变量名,而显示出错信息。C语言认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加 可读性。2在程序嵌套比较多时,容易出错在程序嵌套比较多时,很容易多卸“ ”,或者少写“ ”,造成编译错误,为减少类似 错误出现,在写程序时应该采取缩进式写法,让处于同一层次的函数处于同一条线上,这 样就能够减少上述错

14、误的出线3. win7不支持C语言图形全屏模式,图形处理须加载图形驱动在进行图形部分处理时 win7不支持c语言图形的全屏模式,在 XP系统下,也没法正 确运行。原因是C语言在进行图形处理必须正确设置图形驱动器和模式,通过函数detectgraphics能自动进行计算机硬件检测,后再进行初始华即可。4. 布雷随机数生成的错误以及优化在进行布雷函数编写时,首先用随机数生成函数使随机数生成范围在(0-1)之间,使得地雷主要分布与上部,下部区域基本没有地雷,为了使得雷区较为平均,我又加了一重 循环,强制每行最多只能分布 2个地雷,并另为生成两个随机数,强制是随机数的行数, 该行不分布地雷,这样使得雷

15、区相对平均,但语句过于繁琐,且对于知道该布雷规则的人,游戏更加容易,最后我们采用了加大随机数生成范围,让随机数在(0, t)之间,t可以调节就是得雷区均匀的同时,可以通过调节t的值来控制某局地雷数目的多少。而在进行生成随机数时时间种子的生成位置会影响随机数的生成,因为在计算机中随机数一旦生成, 计算机就不会再生成随机数,会导致生成的随机数相同。5在调用递归函数及递归函数编写时的错误在调用递归函数前应该先调用 Mi neStatistics函数统计该格子周围是否无雷,只有当该 格子无雷且周围无雷时(而不时该格子处无雷即可),才能调用递归函数向八个方向递归展 开无雷区域,该递归函数的出口是碰到了已

16、经处理过的格子或者该格子已经做了标记,又 或者是该格子有雷,而在进行递归在向周围的八个方向展开时要注意设置边界值的情况, 否则会造成处理数据超出数组定义的边界,造成不必要的错误6. Min eStatistics函数编写时条件不易设置函数Min eStatistics编写时不同情况进行分别统计时条件设置很容易错误,类似于鞍 点查找时的错误,导致一类点周围的雷数的统计出现错误,最后我采用在纸上画出模型, 在进行某类点周围雷数统计编写时,根据纸上模型的规律进行编写7输赢判断是靠检索非雷的格子是否处理完游戏的输赢判断有两种思路,一是检索是雷的格子是否已经全部做处理(即标红旗),二是检索非雷的格子是否

17、已经全部处理完全。雷的数目虽然较少,但是雷点具有不确定性,为了实现方法一的判断,还需定义一个二维数组,用于保存哪些位置是有地雷分布的,在进行判断时,首先读取该数组的标记,若该格子有雷,再从结构体数组中读取flag的值,看该格子的地雷是否已经做了处理。 而,进行方法二的实现只需要每处理一个非雷的格子,让统计非雷数目的数据自减一,当其自减为 0时,则表示玩家取得了胜利,比较两种处理 方法,知道方法二比较容易实现8.该函数鼠标的光标没法显示出来该函数鼠标的光标没法显示出来, 在查询了关鼠标光标显示后, 进行了鼠标光标显 示的尝试。但出现了光标只有在鼠标单击后才能显示,通过条件设置使得能在移动时显示

18、鼠标的位置,但鼠标光标的图形会遮盖原有图形的显示。该进方法有两种,一是每次进行 鼠标显示后进行清屏处理,但这样需要在清屏前保存所有位置点的信息,数据的保存,读 取较复杂,没有实现;二是,在每次进行鼠标光标输出前通过数据记录改点的图形信息, 在输出鼠标,且鼠标移出改点后,用原来的图形信息通过图形输出覆盖掉鼠标光标显示留 下的痕迹还原原有图形,该方法的实现也较为复杂,亦没有实现。由于该部分功能没有实 现,没有将本部分相关的代码附录出来六附录#in clude#in clude#in clude#in clude#defi ne LEFTPRESS OxffOl#defi ne LEFTCLICK

19、0xff10#defi ne LEFTDRAG 0xff19#defi ne MOUSEMOVE 0xff08void Ini t();void Close();void DrawSmile();void DrawRedFlag(i nt i,i nt j);void DrawEmpty(int i,int j,int mode,int color); void Con trol();intMineStatistics(int i,int j);void SpreadMi ne(i nt t); void ShowWhite(i nt i,i nt j); void GamePlay();vo

20、id GameBega in();void GameWi n();void GameOver();void Mouse On ();void MouseOff();void MouseSetXY(i nt x,i nt y); int LeftPress();int RightPress(); void MouseGetXY(); char randminenu m3; int minenum;int MouseX,MouseY;int gameAga in=0;int gamePlay=0; structint num;int roundnum;格内的雷数*/int flag;Mi ne10

21、10;mai n()In it();/*无雷标0,有雷标1*/*此位置有雷标0,否则统计周围九个方/*标记为雷时为1,没有标记为0*/Con trol();Close(); getch();void Mouse On()_AX=0x01; ge nin terrupt(0x33);void MouseOff()/*鼠标光标显示*/*鼠标光标隐藏*/_AX=0x02; voidMouseSetXY(i nt x,i nt y)/*设置鼠标当前位置*/_CX=x;_DX=y; _AX=0x04;ge nin terrupt(0x33);int LeftPress()_AX=0x03;ge nin

22、terrupt(0x33); return(_BX &1);int RightPress()/*鼠标左键按下*/*鼠标右键按下*/_AX=0x03;ge nin terrupt(0x33); return(_BX&2); void MouseGetXY()_AX=0x03;ge nin terrupt(0x33);MouseX=_CX;MouseY=_DX; _/*得到当前位置*/ge nin terrupt(0x33);void In it()/*图形初始化*/int gdriver, gmode;registerbgidriver(EGAVGA_driver); /* 加载图形驱动包 */

23、 detectgraph( & gdriver, &gmode);/*检测硬件确定图形驱动程序和模式 */initgraph(&gdriver, &gmode, );/* 初始化图形系统,gdriver 和 gmode 分别表示图形驱动器和模式,内为图形驱动的路径*/void DrawSmile()/* 花笑脸 */ setfillstyle(SOLID_FILL,YELLOW); /*设置填充模式和颜色,表式对下面所画图形的填充,此处表示对下面椭圆的填充*/fillellipse(290,75,10,10); BW/*画椭圆,前两个参数表示中心点坐标,后面依 次为一个宽度和一个高度*/set

24、color(YELLOW);(卜设置画笔颜色*/setfillstyle(SOLID_FILL,BLACK);/*此处表示对下面两个椭圆的填充*/fillellipse(285,75,2,2); fillellipse(295,75,2,2); setcolor(BLACK);bar(287,80,293,81);void DrawRedFlag(i nt i,i nt j)setcolor(7); setfillstyle(SOLID_FILL,RED);/*画一个二维线条*/*设置画笔颜色*/bar(198+j*20,95+i*20,198+j*20+5,95+i*20+5);setcol

25、or(BLACK);lin e(198+j*20,95+20*i,198+j*20,95+i*20+5); void DrawEmpty(i nt i,i nt j,i nt mode,i nt color) setcolor(color);setfillstyle(SOLID_FILL,color);if(mode=0)/*没有单击过显示大格子*/bar(200+j*20-8,100+i*20-8,200+j*20+8,100+i*20+8);if(mode=1)/*单击过后显示小格子*/bar(200+j*20-7,100+i*20-7,200+j*20+7,100+i*20+7);voi

26、d Close()closegraph();void SpreadMi ne(i nt t)int i,j;minenum=0;sran d(i nt)time(0);for(i=0;i10;i+) for(j=0;j10;j+)Mineij.num=rand()%t;ftE.lv/*生成时间种子*/if(Mi neij. nu m=1)minenu m+;elseMin eij. num=2;Min eij.flag=O;sprintf(randminenum,%d, minenum);setcolor(l);settextstyle(0,0,2);outtextxy(210,70,ra n

27、dmi nenu m);minenum=100-minenum;/*变量取空白格数量*/Mouse On ();void GameOver()int i,j;setcolor(0);for(i=0;i10;i+)for(j=0;j10;j+)if(Mi neij. nu m=1)DrawEmpty(i,j,0,RED); setfillstyle(SOLID_FILL,BLACK); fillellipse(200+j*20,100+i*20,7,7);void GameWi n()setcolor(1);settextstyle(0,0,2); outtextxy(230,30,You Wi

28、n);int Min eStatistics(i nt i,i nt j) int n NUM=0;if(i=0&j=0)/*统计左上角*/if(Mi ne01. num=1) n NUM+;if(Mi ne10. nu m=1) n NUM+;if(Mi ne11. num=1) n NUM+;VSKbS jelse if(i=0&j=9)/*统计右上角*/if(Mi ne18. nu m=1) n NUM+;else if(i=9&j=0)if(Mi ne91. num=1) n NUM+;if(Mi ne80. nu m=1) n NUM+;if(Mi ne81. num=1) n NU

29、M+;else if(i=9&j=9)if(Mi ne98. nu m=1) n NUM+;if(Mi ne89. nu m=1) n NUM+;if(Mi ne88. nu m=1) n NUM+;else if(i=0&j!=0&j!=9)if(Mi neij+1 .n um=1) n NUM+;if(Mi neij-1. nu m=1) n NUM+;if(Mi nei+1j-1. num=1) n NUM+;if(Mi nei+1j. num=1) n NUM+;if(Mi nei+1j+1. num=1) n NUM+;else if(i=9&j!=0&j!=9)if(Mi neij

30、+1 .n um=1) n NUM+;if(Mi neij-1. nu m=1) n NUM+;/*统计左下角*/*统计右下角*/*统计第一行*/*统计最后一行*/if(Mi nei-1j+1. num=1) n NUM+;else if(j=0&i !=0&i!=9)/* 统计第一列 */if(Mi nei-1j. num=1) n NUM+;if(Mi nei+1j. num=1)n NUM+;if(Mi neij+1 .n um=1) n NUM+;if(Mi nei-1j+1. num=1) n NUM+;if(Mi nei+1j+1. num=1) n NUM+;else if(j=

31、9&i!=0&i!=9)/* 统计最后一列 */if(Mi nei-1j. num=1) n NUM+;if(Mi nei+1j. num=1)n NUM+;if(Mi neij-1. nu m=1)n NUM+;if(Mi nei-1j-1. nu m=1) n NUM+;if(Mi nei+1j-1. num=1)n NUM+;elseif(Mi nei-1j-1. nu m=1) n NUM+;if(Mi nei-1j. num=1)n NUM+;/fif(Mi nei-1j+1. nu m=1) n NUM+;if(Mi neij-1. num=n NUM+;if(Mi neij+1 .n um=1)n NUM+;if(Mi nei+1j+1. num=1) n NUM+; return( nNUM);void GameBega in()int i,j;cleardevice(); if(gamePlay!=1) MouseSetXY(290,70);MouseX=290; M

温馨提示

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

评论

0/150

提交评论