失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > 99行不用指针 c语言贪食蛇 小白99行C代码编写经典游戏贪吃蛇(多关版本)

99行不用指针 c语言贪食蛇 小白99行C代码编写经典游戏贪吃蛇(多关版本)

时间:2021-08-24 11:39:58

相关推荐

99行不用指针 c语言贪食蛇 小白99行C代码编写经典游戏贪吃蛇(多关版本)

文章目录

前言

一、实现原理

1、地图打印

2、蛇身运动

3、方向键操控

4、四种判定

5、函数工具

二、游戏试玩

1、编译环境

2、源代码

三、代码详解

1、游戏配置部分(第1~31行)

2、游戏主体部分(第32~80行)

3、游戏结束部分(第81~99行)

结语

前言

学完C语言但不会写贪吃蛇?

嗯嗯没毛病,书上的C语言其给人的感觉也就是用来输出设计好的文字和图像,或者拿来解决数学问题,一顿操作之后输出函数静静地在屏幕上输出结果。而这些都是静态的输出,但游戏应该是动态的输出。

所以要编写贪吃蛇,首先要解决的就是如何输出动态图像的问题。

一、实现原理

二维数组和链表?并没有用到,连开始用的结构体都在后期优化当中用两个一维数组代替了。

1、地图打印

二维数组存放地图打印刷新烦人,地图越大越晃眼。这里采用坐标函数配合printf()函数来打印地图,并且打印蛇身和食物也是用这个方法。

使用坐标函数需要调用头文件windows.h,然后自定义这个函数,名字叫什么都可以,这里沿用gotoxy()。其作用很简单,就是将光标移动到指定的坐标。

void gotoxy(int x,int y)

{

COORD p;

p.X=x,p.Y=y;

SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE),p);

}

gotoxy()的详细函数说明看这里。

两个for循环加gotoxy()就可轻松打印出方形地图,并且只需打印一次,不像二维数组一样要重复刷新。

注意我编写代码用的是汉字作为输出图形,一个汉字占两个字符位置,从设计的坐标位置到打印坐标位置横坐标要乘以2。

2、蛇身运动

只需用坐标打印新蛇头并删除蛇尾就能完成一次移动,随后让蛇身的后一节坐标依次等于前一节坐标,完成下次移动的准备。

整个蛇身的移动由循环函数实现,以Sleep()函数暂停以实现速度控制,而不是瞬间完成循环。

3、方向键操控

由kbhit()、getch()和switch()函数完成实现:敲击方向键,退出当前方向运动循环,重新进入键入方向对应的运动循环。

4、四种判定

撞墙判定:由蛇头部和墙坐标重合触发,退出运动循环进而选择重新开始或退出游戏。

自食判定:由蛇头部和蛇身坐标重合触发,退出运动循环进而选择重新开始或退出游戏。

增长判定:由蛇头部和食物坐标重合触发,然后蛇身增长,食物重新刷新。

通关判定:由蛇总体长度达到设定长度触发,退出运动循环进而选择下一难度或退出游戏。

5、函数工具

gotoxy()、srand()、rand()、Sleep()、kbhit()、getch(),system(),整个实现游戏的代码需要用到以上7个函数,其他的就只是for()、while()、if()、switch()等基本函数了。

其中srand()函数和rand()函数仅用于食物的随机刷新,srand()函数仅出现1次,rand()函数出现于2处。

kbhit()函数和getch()函数用于键盘方向键操作和游戏结束时的选择,kbhit()函数出现3次,getch()函数出现2次。

Sleep()函数仅出现1次。

system(“cls”)用于重启游戏时刷屏,仅出现1次。

这7个函数的讲解可查看我的其他博文,或自行搜索。

二、游戏试玩

1、编译环境

编码选择:建议选择GB18030,否则游戏中的汉字可能出现乱码。或者也可以选择把代码中的汉字重打一遍。

游戏源代码编译通用性检测:用了一新一老编译器测试,通用性良好,毕竟vc++6.0都能编译通过。

2、源代码

#include

#include

#include

#include

#include

int I=13,J=16,L=6,level=1,_speed=700;

void gotoxy(int x,int y){

COORD p;

p.X=x,p.Y=y;

SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE),p);

}

int main()

{

res:int x[100],y[100],xo,yo,choose=2,length=3,fact=0,ability=0;

x[1]=3,y[1]=6,x[2]=2,y[2]=6,x[3]=1,y[3]=6;

for(int i=0;i

gotoxy(0,i);

printf("田");

gotoxy(2*(J-1),i);

printf("田");

}

for(int j=0;j

gotoxy(2*j,0);

printf("田");

gotoxy(2*j,I-1);

printf("田");

}

srand((unsigned) time(0));

xo=rand()%(J-2)+1,yo=rand()%(I-2)+1;

gotoxy(2*xo,yo);

printf("虫");

while(1){

if(kbhit())

switch(getch()){

case 75:choose=1;break;

case 77:choose=2;break;

case 72:choose=3;break;

case 80:choose=4;

}

while(1){

switch(choose){

case 1:x[0]=x[1]-1,y[0]=y[1];break;

case 2:x[0]=x[1]+1,y[0]=y[1];break;

case 3:y[0]=y[1]-1,x[0]=x[1];break;

case 4:y[0]=y[1]+1,x[0]=x[1];

}

if(kbhit())break;

if(x[0]==xo && y[0]==yo){

xo=rand()%(J-2)+1,yo=rand()%(I-2)+1;

length++;

gotoxy(34,0);

printf("难度等级:%2d 进度得分:%3d",level,100*(length-3)/(L-3));

}

else{

gotoxy(2*x[length],y[length]);

printf(" ");

}

gotoxy(2*x[0],y[0]);

printf("口");

if(length==L)break;

for(int k=2;k<=length;k++)

if(x[0]==x[k] && y[0]==y[k])fact=1;

if(fact)break;

gotoxy(2*xo,yo);

printf("虫");

gotoxy(30,20);

Sleep(_speed);

for(int i=length;i>1;i--)x[i]=x[i-1],y[i]=y[i-1];

switch(choose){

case 1:x[1]--;break;

case 2:x[1]++;break;

case 3:y[1]--;break;

case 4:y[1]++;

}

if(x[1]<1||x[1]>J-2||y[1]<1||y[1]>I-2) break;

}

if(x[1]<1||x[1]>J-2||y[1]<1||y[1]>I-2) break;

if(fact)break;

if(length==L)break;

}

if(length==L){

gotoxy(11,14);

printf("恭喜通关!");

ability=1,level++;

printf("\n输入1进入下一难度,输入2退出游戏:");

}

else{

gotoxy(11,14);

printf("GAME OVER!");

printf("\n输入1重启游戏,输入2退出游戏:");

}

sty:while(1)if(kbhit())break;

switch(getch()){

case 49:if(ability==1)_speed=0.6*_speed,L=L/0.6;system("cls");goto res;

case 50:break;

default:goto sty;

}

return 0;

}

游戏截图:

三、代码详解

1、游戏配置部分(第1~31行)

第6行:int I=13,J=16,L=6,level=1,_speed=700;声明全局变量:

I和J是设定地图的宽和长;

L是通关长度;

level是难度等级;

_speed是Sleep()函数的参数,初始设定700毫秒。

第7~11行:定义坐标函数gotoxy(),因为全程要用到,所以定义在了主函数之前,还能省一行声明,默默地给机智的自己点一个赞。

第14行:res:int x[100],y[100],xo,yo,choose=2,length=3,fact=0,ability=0;声明变量:x[100]和y[100]是存放蛇身坐标的数组;

xo和yo是食物的坐标;

choose是选择运动方向的辅助变量,设定值为2表示开始时自动向右行进;

length是初始蛇长;

fact是辅助自食判定的变量;

ability是辅助判定是否提升难度的变量;

res:配合goto语句重启游戏。

第15行:初始化前三节蛇身坐标。

第16~27行:打印地图。

第28~31行:打印第一次的食物。

2、游戏主体部分(第32~80行)

第32行和第40行的while(1)构成嵌套循环游戏主体。

第33~39行:敲击键盘就进行方向选择,不敲击就跳过。75、77、72、80是左右上下的双ASCII码之一。

第41~46行:对应运动方向,把新蛇头坐标赋值给x[0]和y[0],如当choose=1时,x[0]=x[1]-1,y[0]=y[1];x[1]-1和y[1]就是新蛇头坐标,也就是向左移动了一格。

第47行:if(kbhit())break;检测到有键盘输入则中断循环回到上一循环,没有就跳过。

第48~57行:检测新蛇头是否吃到食物:

吃到就重新随机食物坐标,蛇身长度变量加1,得分增加,蛇尾不删除;

没吃到就去蛇尾坐标位置把蛇尾删掉。

第58~59行:打印新蛇头。

第60行:if(length==L)break;通关判定:

长度达标则跳出循环,配合第79行的相同判定跳出双嵌套循环进入游戏结束阶段;

没达标就过。

第61~63行:自食判定:

新蛇头坐标和蛇身坐标重合则跳出循环,配合第78行if(fact)break;再次判定跳出双嵌套循环进入游戏结束阶段;

没自食就过。

第64~66行:打印食物,移开光标。

第67行:Sleep(_speed);速度控制。

第68行:for(int i=length;i>1;i--)x[i]=x[i-1],y[i]=y[i-1];蛇身各坐标顺序向前一格。(仅赋值,下一循环才打印)

第69~74行:单独移动蛇头坐标,之前x[0]和y[0]是提前用了新蛇头坐标,在这里才将x[1]或y[1]移动,放在最后是因为第68行的蛇身赋值要求蛇头坐标最后改变。

第75行:if(x[1]<1||x[1]>J-2||y[1]<1||y[1]>I-2) break;撞墙判定:

撞了就跳出循环,配合第77行的相同判定跳出双嵌套循环进入游戏结束阶段;

没撞就过;

3、游戏结束部分(第81~99行)

第81~91行:判断游戏通关还是游戏失败:

长度达标则输出“恭喜通关”和选择重来或退出的提示,ability赋值为1,等级提升;

长度未达标跳出主体循环就是游戏失败了,输出“GAME OVER”和选择重来或退出的提示。

第92~97行:while(1)等待,有键盘输入则跳出并继续。49和50是1和2的ASCII码。选1进入下一难度,速度加快,通关长度要求增加,清屏回到初始代码位置,选2退出;其他键不处理,重新等待。

结语

要是超过100个赞我就考虑再写推箱子好还是俄罗斯方块好。

整个代码还有2行是不必要的,有兴趣的可以找找看。

如果觉得《99行不用指针 c语言贪食蛇 小白99行C代码编写经典游戏贪吃蛇(多关版本)》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。