c语言课程设计扫雷.docx
- 文档编号:5039648
- 上传时间:2022-12-12
- 格式:DOCX
- 页数:23
- 大小:483.32KB
c语言课程设计扫雷.docx
《c语言课程设计扫雷.docx》由会员分享,可在线阅读,更多相关《c语言课程设计扫雷.docx(23页珍藏版)》请在冰豆网上搜索。
c语言课程设计扫雷
c语言课程设计——扫雷
安徽工业大学
C语言程序设计报告
题目:
扫雷游戏
班级:
网101
组长:
~~
成员:
~~,~~
日期:
目录
一基本情况
二游戏规则
三设计思路
四函数说明
课程设计
1.游戏控制部分
2.图形处理部分
3.鼠标处理部分
五调试与测试
六附录
一基本情况
开发平台:
win-tc(windows-xp)
开发进度:
~小组进行图形处理及鼠标相关资料收集,掌握相关图形处理及鼠标相关技术
~小组成员讨论程序实现算法及任务分配
~代码实现,调试
任务划分:
陈超负责程序总体设计,游戏控制部分实现,后期调试
张艳负责程序图形处理及鼠标处理部分实现,总结文档编写
课程设计
二游戏规则
课程设计
玩家进入游戏界面后,只能通过鼠标进行操作,一旦有键盘输入,则退出游戏程序。
游戏框图左上方显示的数字为此局游戏中包含的地雷数目。
中间黄色笑脸为重新开始按钮,一旦鼠标左键单击此位置,则退出正在进行的扫雷游戏,游戏重新开始。
游戏中包含10*10个空格,可以通过鼠标左键单击或者右键单击来进行扫雷游戏。
鼠标单击表示打开此空格位置,若此位置为地雷,则输出所有的地雷分布,玩家输,游戏结束,等待玩家的选择退出或者继续开始下一次游戏。
若此位置已经被左单击过,则再次进行鼠标左键单击时,游戏不做任何响应。
单击过后若此位置无雷则显示空格或者数字。
空格表示此处无雷,数字表示此处无雷且周围八个格子中的地雷数目为空格中的数字数目,若某个空格周围亦没有地雷分布则继续以每个方格为中心继续判断无雷就展开,显示数字或者空格,直至遇到地雷或者是边界。
鼠标右键单击表示游戏玩家标识此位置为地雷,若游戏玩家在某位置右键单击过之后,再次右键单击此位置,则表示取消此处标注的地雷标志,若在此处鼠标左键单击,不做任何响应。
游戏玩家依次对每一个空格,当游戏玩家把所有的非雷区域全部处理完毕,则游戏结束,玩家胜利,输出“YOUWIN”,等待玩家进一步选择。
三设计思路
扫雷程序主要用了一个10行10列的二维数组,数组的每个元素都是结构体类型,结构体中的num代表格内当前处于什么状态,值1表示有雷,值0表示已经变成空白格或者显示过数字,roundnum统计每个格子周围有多少地雷,flag是鼠标右键的标志,如果鼠标左键单击此方格,则对应的二维数组结构体里的flag为1表示格子显示红旗,这样鼠标左键点击在这个格子上无效。
首先通过生成随机数的方式布置地雷,如果在该点生成的随机数为1,则该点存在地雷,将NUM(结构体)标为1,随机数为其它值NUM(结构体)标为0,每生成一个地雷,则全局变量minenum自增一。
课程设计
在程序一开始统计好每个格子周围有多少地雷,将统计的结果存放在对应的结构体变量roundnum,然后当鼠标左键点击在没有地雷的格子上时进行两种判断,如果格子周围没有地雷,就在原来的格子位置显示空白格,然后用递归的方法同样判断周围的8个格子;如果格子周围有地雷,就在该空格显示具体的雷数。
在递归判断8个格子时,如果格子上有雷或者格子已经显示过雷数或者空白,以及格子上有红旗标志的话,就不再对格子进行任何判断。
否则,当此处无雷,即展开此处,并且以此位置为中心继续进行递归,直至遇到边界或者地雷,跳出递归。
当游戏玩家每处理一个非雷区域的格子,则对应的统计非雷方格数目的变量自减一,当该变量的值自减为0时,则表示游戏结束,该玩家胜利。
具体函数流程图如下:
课程设计
四函数说明
1游戏控制部分
A.main()主函数
定义使用到的常数,全局变量,函数原型说明。
然后初始化图形系统,调用游戏控制函数,按任意键结束游戏,关闭图形系统,游戏结束。
B.voidSpreadMine(int t)布雷函数
通过函数srand((int)time(0));生成随机数,随后在二重循环里,生成随机数,若生成的随机数为1,则将num(结构体)标为1,表示改点有雷,并在该函数内统计该次游戏中地雷总数,通过outtextxy(210,70,randminenum);在(210,70)点输出雷的总数,用一百减去总雷数表示该次无雷空格的总数(在输赢判断中,若所有无雷区域都已经得到了处理,即玩家胜利)。
C.voidGameOver()游戏结束
课程设计
如果单击了地雷,则地雷爆炸,游戏结束,显示地雷分布,两重循环判断每个格子的状态,如果状态值为1,则为地雷,设置红色背景,实体填充模式,画圆表示地雷。
D.voidGameWin()玩家胜利
如果所有的非雷区域的格子都处理过,就表示游戏玩家胜利,输出youwin信息。
GameBegain()游戏开始
这个函数完成初始界面的设计以及调用函数voidSpreadMine(int t)生成地雷。
初始界面的主要工作是确定图的位置和方格显示的位置。
外边框的左上角为(190,60),右上角坐标为(390,290),显示笑脸和地雷数的区域为(190,60)~(390,90),每个方格的宽度和高度均为16.
F.voidShowWhite(inti,intj)显示空白区域
当单击某个格子,该格子周围的地雷数为0时,应该继续判断它周围8个方向的格子的周围的雷数是否为0,将这些格子都用空白显示。
具体实现如下:
a如果当前格子有红旗或者已经显示过数字或空白格子,则返回。
b空白格子数减一(统计处理过多少格子)。
c如果周围的地雷数为0,且它不是地雷,则将它显示为空白,同时将它的状态值num置0,表示处理过;如果周围地雷数不为0,显示周围地雷数,同时将它的状态值num置0,表示处理过,返回。
G.voidControl()
调用函数GameBegain()显示游戏初始化界面,调用GamePlay()函数具体实现游戏操作,该函数的返回值有可能有两种:
一是正在玩中,提前结束游戏;一是玩完,要么失败,要么胜利。
如果是前者,则重新开始游戏。
如果是后者,则判断是否单击了笑脸,是则重新开始,否则结束程序。
课程设计
H.voidGamePlay()游戏控制
游戏过程主要是对鼠标按键的处理,集体算法实现如下:
a.如果单击了鼠标左键则判断
如果单击了笑脸,则游戏重新开始‘
如果单击了方格,判断该格子有红旗,则按键无效。
如果单击的格子没有显示过数字或空白,则判断:
如果是地雷,游戏结束,显示地雷分布;如果不是地雷,则统计周围(8个方向)的地雷数,如果周围地雷数为0,条用递归函数ShowWhite()处理周围格子的情况,显示周围的格子数或者空白。
如果单击的格子周围地雷数不为0,则显示周围地雷数,将处理过的格子做标记。
如果处理完所有的格子吗,则游戏胜利,显示胜利信息。
b.如果单击鼠标右键,该格子没有红旗则显示红旗。
如果有红旗,再击右键,则红旗消失。
c重复步骤b直到按键结束游戏
I.intMineStatistics(inti,intj)
课程设计
统计每个格子周围的地雷数,分别考虑各自处于四个角,四条边以及中间某个位置的情况。
周围是上,下,左,右,左上,左下,右上,右下。
在统计四个角格子周围的地雷数目时,统计其周围的三个方格内是否有地雷,如统计左上角位置周围的地雷数目时,其周围的三个方格的二维坐标依次为(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+1,j+1).统计完成后,将统计结果存入对应的结构体的roundnum变量中去。
2.图形处理部分
A.voidInit()对计算机进行图形模式初始化,加载图形驱动包
B.voidDrawSmile()画一个笑脸
C.voidDrawRedFlag(inti,intj)画一个红旗
D.voidDrawEmpty(inti,intj,intmode,intcolor)画两种空格
E.
课程设计
voidClose()关闭图形模式
3.鼠标处理部分
A.voidMouseOn()鼠标显示
B.voidMouseOff()鼠标隐藏
C.voidMouseSetXY(intx,inty)设置鼠标当前位置
D.intLeftPress()鼠标左键按下
E.intRightPress()鼠标右键按下
课程设计
F.voidMouseGetXY()得到鼠标当前位置
五调试与测试
1.书写标识符时,忽略了大小写字母的区别
编译程序把i和I认为是两个不同的变量名,而显示出错信息。
C语言认为大写字母和小写字母是两个不同的字符。
习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
2.在程序嵌套比较多时,容易出错
在程序嵌套比较多时,很容易多卸“{”,或者少写“{”,造成编译错误,为减少类似错误出现,在写程序时应该采取缩进式写法,让处于同一层次的函数处于同一条线上,这样就能够减少上述错误的出线
3.win7不支持C语言图形全屏模式,图形处理须加载图形驱动
课程设计
在进行图形部分处理时win7不支持c语言图形的全屏模式,在XP系统下,也没法正确运行。
原因是C语言在进行图形处理必须正确设置图形驱动器和模式,通过函数detectgraphics能自动进行计算机硬件检测,后再进行初始华即可。
4.布雷随机数生成的错误以及优化
在进行布雷函数编写时,首先用随机数生成函数使随机数生成范围在(0-1)之间,使得地雷主要分布与上部,下部区域基本没有地雷,为了使得雷区较为平均,我又加了一重循环,强制每行最多只能分布2个地雷,并另为生成两个随机数,强制是随机数的行数,该行不分布地雷,这样使得雷区相对平均,但语句过于繁琐,且对于知道该布雷规则的人,游戏更加容易,最后我们采用了加大随机数生成范围,让随机数在(0,t)之间,t可以调节就是得雷区均匀的同时,可以通过调节t的值来控制某局地雷数目的多少。
而在进行生成随机数时时间种子的生成位置会影响随机数的生成,因为在计算机中随机数一旦生成,计算机就不会再生成随机数,会导致生成的随机数相同。
5在调用递归函数及递归函数编写时的错误
在调用递归函数前应该先调用MineStatistics函数统计该格子周围是否无雷,只有当该格子无雷且周围无雷时(而不时该格子处无雷即可),才能调用递归函数向八个方向递归展开无雷区域,该递归函数的出口是碰到了已经处理过的格子或者该格子已经做了标记,又或者是该格子有雷,而在进行递归在向周围的八个方向展开时要注意设置边界值的情况,否则会造成处理数据超出数组定义的边界,造成不必要的错误
6.MineStatistics函数编写时条件不易设置
函数MineStatistics编写时不同情况进行分别统计时条件设置很容易错误,类似于鞍点查找时的错误,导致一类点周围的雷数的统计出现错误,最后我采用在纸上画出模型,在进行某类点周围雷数统计编写时,根据纸上模型的规律进行编写
7.输赢判断是靠检索非雷的格子是否处理完
课程设计
游戏的输赢判断有两种思路,一是检索是雷的格子是否已经全部做处理(即标红旗),二是检索非雷的格子是否已经全部处理完全。
雷的数目虽然较少,但是雷点具有不确定性,为了实现方法一的判断,还需定义一个二维数组,用于保存哪些位置是有地雷分布的,在进行判断时,首先读取该数组的标记,若该格子有雷,再从结构体数组中读取flag的值,看该格子的地雷是否已经做了处理。
而,进行方法二的实现只需要每处理一个非雷的格子,让统计非雷数目的数据自减一,当其自减为0时,则表示玩家取得了胜利,比较两种处理方法,知道方法二比较容易实现
8.该函数鼠标的光标没法显示出来
该函数鼠标的光标没法显示出来,在查询了有关鼠标光标显示后,进行了鼠标光标显示的尝试。
但出现了光标只有在鼠标单击后才能显示,通过条件设置使得能在移动时显示鼠标的位置,但鼠标光标的图形会遮盖原有图形的显示。
该进方法有两种,一是每次进行鼠标显示后进行清屏处理,但这样需要在清屏前保存所有位置点的信息,数据的保存,读取较复杂,没有实现;二是,在每次进行鼠标光标输出前通过数据记录改点的图形信息,在输出鼠标,且鼠标移出改点后,用原来的图形信息通过图形输出覆盖掉鼠标光标显示留下的痕迹还原原有图形,该方法的实现也较为复杂,亦没有实现。
由于该部分功能没有实现,没有将本部分相关的代码附录出来
课程设计
六附录
#include<>
#include<>
#include<>
#include<>
#defineLEFTPRESS0xff01
#defineLEFTCLICK0xff10
#defineLEFTDRAG0xff19
#defineMOUSEMOVE0xff08
voidInit();
voidClose();
voidDrawSmile();
voidDrawRedFlag(inti,intj);
voidDrawEmpty(inti,intj,intmode,intcolor);
voidControl();
intMineStatistics(inti,intj);
voidSpreadMine(intt);
voidShowWhite(inti,intj);
voidGamePlay();
课程设计
voidGameBegain();
voidGameWin();
voidGameOver();
voidMouseOn();
voidMouseOff();
voidMouseSetXY(intx,inty);
intLeftPress();
intRightPress();
voidMouseGetXY();
charrandminenum[3];
intminenum;
intMouseX,MouseY;
intgameAgain=0;
intgamePlay=0;
struct
{
intnum;/*无雷标0,有雷标1*/
introundnum;/*此位置有雷标0,否则统计周围九个方格内的雷数*/
intflag;/*标记为雷时为1,没有标记为0*/
}Mine[10][10];
main()
{
Init();
Control();
Close();
getch();
}
voidMouseOn()/*鼠标光标显示*/
{
_AX=0x01;
geninterrupt(0x33);
}
voidMouseOff()/*鼠标光标隐藏*/
课程设计
{
_AX=0x02;
geninterrupt(0x33);
}
voidMouseSetXY(intx,inty)/*设置鼠标当前位置*/
{
_CX=x;
_DX=y;
_AX=0x04;
geninterrupt(0x33);
}
intLeftPress()/*鼠标左键按下*/
{
_AX=0x03;
geninterrupt(0x33);
return(_BX&1);
}
intRightPress()/*鼠标右键按下*/
{
_AX=0x03;
geninterrupt(0x33);
return(_BX&2);
}
voidMouseGetXY()/*得到当前位置*/
{
_AX=0x03;
geninterrupt(0x33);
MouseX=_CX;
MouseY=_DX;
}
voidInit()/*图形初始化*/
{
intgdriver,gmode;
registerbgidriver(EGAVGA_driver);/*加载图形驱动包*/
detectgraph(&gdriver,&gmode);/*检测硬件确定图形驱动程序和模式*/
initgraph(&gdriver,&gmode,"");/*初始化图形系统,gdriver和gmode分别表示图形驱动器和模式,""内为图形驱动的路径*/
}
课程设计
voidDrawSmile()/*花笑脸*/
{
setfillstyle(SOLID_FILL,YELLOW);/*设置填充模式和颜色,表式对下面所画图形的填充,此处表示对下面椭圆的填充*/
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);/*画一个二维线条*/
}
voidDrawRedFlag(inti,intj)
{
setcolor(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);
}
voidDrawEmpty(inti,intj,intmode,intcolor)
{
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);
}
voidClose()
{
closegraph();
}
voidSpreadMine(intt)
{
inti,j;
minenum=0;
srand((int)time(0));/*生成时间种子*/
for(i=0;i<10;i++)
课程设计
for(j=0;j<10;j++)
{
Mine[i][j].num=rand()%t;
if(Mine[i][j].num==1)
minenum++;
else
Mine[i][j].num=2;
Mine[i][j].flag=0;
}
sprintf(randminenum,"%d",minenum);
setcolor
(1);
settextstyle(0,0,2);
outtextxy(210,70,randminenum);
minenum=100-minenum;/*变量取空白格数量*/
MouseOn();
}
voidGameOver()
{
inti,j;
setcolor(0);
for(i=0;i<10;i++)
for(j=0;j<10;j++)
if(Mine[i][j].num==1)
{
DrawEmpty(i,j,0,RED);
setfillstyle(SOLID_FILL,BLACK);
fillellipse(200+j*20,100+i*20,7,7);
}
}
voidGameWin()
{
setcolor
(1);
settextstyle(0,0,2);
outtextxy(230,30,"YouWin");
}
intMineStatistics(inti,intj)
{
intnNUM=0;
if(i==0&&j==0)/*统计左上角*/
{
if(Mine[0][1].num==1)
nNUM++;
if(Mine[1][0].num==1)
课程设计
nNUM++;
if(Mine[1][1].num==1)
nNUM++;
}
elseif(i==0&&j==9)/*统计右上角*/
{
if(Mine[0][8].num==1)
nNUM++;
if(Mine[1][9].num==1)
nNUM++;
if(Mine[1][8].num==1)
nNUM++;
}
elseif(i==9&&j==0)/*统计左下角*/
{
if(Mine[9][1].num==1)
nNUM++;
if(Mine[8][0].num==1)
nNUM++;
if(Mine[8][1].num==1)
nNUM++;
}
elseif(i==9&&j==9)/*统计右下角*/
{
if(Mine[9][8].num==1)
nNUM++;
if(Mine[8][9].num==1)
nNUM++;
if(Mine[8][8].num==1)
nNUM++;
}
elseif(i==0&&j!
=0&&j!
=9)/*统计第一行*/
{
if(Mine[i][j+1].num==1)
nNUM++;
if(Mine[i][j-1].num==1)
nNUM++;
if(Mine[i+1][j-1].num==1)
nNUM++;
if(Mine[i+1][j].num==1)
nNUM++;
课程设计
if(Mine[i+1][j+1].num==1)
nNUM++;
}
elseif(i==9&&j!
=0&&j!
=9)/*统计最后一行*/
{
if(Mine[i][j+1].num==1)
nNUM++;
if(Mine[i][j-1].num==1)
nNUM++;
if(Mine[i-1][j-1].num==1)
nNUM++;
if(Mine[i-1][j].num==1)
nNUM++;
if(Mine[i-1][j+1].num==1)
nNUM++;
}
elseif(j==0&&i!
=0&&i!
=9)/*统计第一列*/
{
if(Mine[i-1][j].num==1)
nNUM++;
if(Mine[i+1][j].num==1)
nNUM++;
if(Mine[i][j+1].num==1)
nNUM++;
if(Mine[i-1][j+1
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 语言 课程设计 扫雷