C语言实现链表贪吃蛇
本文实例为大家分享了C语言实现贪吃蛇的具体代码,供大家参考,具体内容如下
用C语言链表写的贪吃蛇(程序设计时做的,做的不好大佬勿喷)
借助游戏内容分析贪吃蛇所需的功能主要包括这几块:
1.移动光标模块
2.打印地图模块和基本规则信息
读取最高分文件
3.打印初始蛇模块
打印时给予蛇的初始移动方向
4.产生食物模块
1)、保证食物在地图内产生
2)、保证食物不能出现在蛇体
5.蛇的生命状态判断模块
1)、撞墙导致死亡
2)、头撞身体部位死亡
6.运行模块
1)、让蛇移动
2)、根据按键来改变蛇的移动方向
3)、对待分数的增加游戏难度的增加
4)、蛇在吃食物后分数的增加
7.结束模块
在遇到撞墙或者撞自己部位死亡时结束程序,并进行分数与历史最高分作比较,最终达到最高分的更新
以下为代码
#include <stdlib.h>
#include <stdio.h>
#include <conio.h>//控制台输入和输出
#include <windows.h>//窗口函数
#include <time.h>
#define W 1//蛇的运动方向W:上 S:下 A: 左 D:右
#define S 2
#define A 3
#define D 4
/*
定义全局变量
*/
typedef struct{
int x;
int y;
}place;//定义坐标结构体
typedef struct ZB{
place data;
struct ZB *next;
}snake;//定义蛇的链表
/*
定义全局链表
*/
snake *head,*p,*q,*h;//
place food;//定义食物坐标
int score=0,bestscore,game_flag=0,ch,sleep=400;//定义得分score死亡判断game_flag方向判断ch蛇的速度sleep
/*
函数声明
*/
void gotoxy(int x,int y);//定位光标
void map_creat();//运用定位函数打印地图
void ini_snack();//随机产生蛇
void cre_food();//随机产生食物
void live_jud_1();//判断自己是否撞墙死亡
void live_jud_2();//判断自己是否撞到自己
void move();//蛇的移动
void rungame();//游戏运行
void gameover();//游戏结束界面
void changch();//改变方向
int color(int c);//改颜色函数
/*
构建定位函数
*/
int color(int c)
{
SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE),c); //更改文字颜色
return 0;
}
void gotoxy(int x,int y)//定位光标
{
COORD pos;
HANDLE handle=GetStdHandle(STD_OUTPUT_HANDLE);
pos.X=x;
pos.Y=y;
SetConsoleCursorPosition(handle,pos);
}
/*
打印地图
*/
void map_creat()//运用定位函数打印地图
{
FILE *fp;//创建一个记录最高分的文件
fp=fopen("score.1","r");
fscanf(fp,"%d",&bestscore);//读取最高分
fclose(fp);
gotoxy(54,26);
printf("Your Best Score : %d\t", bestscore);//打印出最高分
int i,j;
for(i=0;i<48;i++)
{
gotoxy(i,0);
printf("#");
color(6);
gotoxy(i,26);
printf("#");
color(6);
}
for(j=0;j<26;j++)
{
gotoxy(0,j);
printf("#");
color(6);
gotoxy(48,j);
printf("#");
color(6);
}
gotoxy(54,10);
printf("游戏规则:");
gotoxy(54,12);
printf("向上移动:↑\n");
gotoxy(54,14);
printf("向下移动:↓\n");
gotoxy(54,16);
printf("向左移动:←\n");
gotoxy(54,18);
printf("向右移动:→\n");
gotoxy(54,20);
printf("吃一个食物分数加10");
gotoxy(54,22);
printf("按空格键暂停游戏");
gotoxy(54,24);
printf("按ESC直接结束游戏");
}
/*
构建初始蛇
*/
void ini_snack()//产生蛇
{
int i;
/*
采用尾插法构建蛇的链表初始长度设为二
*/
head=(snake *)malloc(sizeof(snake));
head->data.x=48/2;
head->data.y=26/2;
head->next=NULL;
h=head;
for(i=1;i<=2;i++)
{
p=(snake *)malloc(sizeof(snake));
p->data.x=48/2+i;
p->data.y=26/2;
h->next=p;
h=p;
}
p->next=NULL;
/*
将蛇打印出来
*/
h=head;
while(h!=NULL)
{
gotoxy(h->data.x,h->data.y);
color(5);
printf("@");
h=h->next;
}
ch=W;//蛇的初始方向
}
/*
随机产生食物
*/
void cre_food()//随机产生食物
{
srand((unsigned)time(NULL));//为了防止每次产生的随机数相同,种子设置为time
/*
随机产生食物
*/
food.x=rand()%(48-2)+1;
food.y=rand()%(26-2)+1;
while(p!=NULL)
{
/*
判断食物是否与蛇重合,如果重合重新产生
*/
if(p->data.x==food.x&&p->data.y==food.y)
cre_food();
p=p->next;
}
gotoxy(food.x,food.y);
color(1);
printf("$");//打印食物
}
/*
判断是否死亡
*/
void live_jud_1()//判断自己是否撞墙死亡
{
if(head->data.x==0||head->data.x==48||head->data.y==0||head->data.y==26)//撞墙
{
game_flag=1;
gameover();
}
}
void live_jud_2()//判断自己是否撞到自己
{
q=head->next;
while(q!=NULL)
{
if(head->data.x==q->data.x&&head->data.y==q->data.y)//撞自己
{
{
game_flag=2;
gameover();
}
break;
}
q=q->next;
}
}
/*
游戏进行界面
*/
void move()
{
snake *l;
live_jud_1();
l=(snake *)malloc(sizeof(snake));
/*
构建一个新的节点通,过新节点来表示下一次头节点所在的位置
将新节点当作移动后的头节点
如果新头节点的坐标等于食物的坐标得分加10,食物的标志变为1
因为蛇的链表长度加了一,如果是移动的话找出尾节点并打印出 然后删掉该尾节点
如果吃到了食物就直接将蛇打印出来
*/
if(ch==W)
{
l->data.x=head->data.x;
l->data.y=head->data.y-1;
if(l->data.x==food.x&&l->data.y==food.y)
{
l->next=head;
head=l;
q=head;
while(q!=NULL)//将蛇重新打印一边
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
score+=10;
cre_food();//构建新的食物
}
else
{
l->next=head;
head=l;
q=head;
while(q->next->next!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
gotoxy(q->next->data.x,q->next->data.y);
printf(" ");//将蛇尾去掉
free(q->next);
q->next=NULL;
}
}
if(ch==A)
{
l->data.x=head->data.x-1;
l->data.y=head->data.y;
if(l->data.x==food.x&&l->data.y==food.y)
{
l->next=head;
head=l;
q=head;
while(q!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
score+=10;
cre_food();
}
else
{
l->next=head;
head=l;
q=head;
while(q->next->next!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
gotoxy(q->next->data.x,q->next->data.y);
printf(" ");
free(q->next);
q->next=NULL;
}
}
if(ch==S)
{
l->data.x=head->data.x;
l->data.y=head->data.y+1;
if(l->data.x==food.x&&l->data.y==food.y)
{
l->next=head;
head=l;
q=head;
while(q!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
score+=10;
cre_food();
}
else
{
l->next=head;
head=l;
q=head;
while(q->next->next!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
gotoxy(q->next->data.x,q->next->data.y);
printf(" ");
free(q->next);
q->next=NULL;
}
}
if(ch==D)
{
l->data.x=head->data.x+1;
l->data.y=head->data.y;
if(l->data.x==food.x&&l->data.y==food.y)
{
l->next=head;
head=l;
q=head;
while(q!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
score+=10;
cre_food();
}
else
{
l->next=head;
head=l;
q=head;
while(q->next->next!=NULL)
{
gotoxy(q->data.x,q->data.y);
color(5);
printf("@");
q=q->next;
}
gotoxy(q->next->data.x,q->next->data.y);
printf(" ");
free(q->next);
q->next=NULL;
}
}
live_jud_2();//判断是否撞自己死亡
}
void rungame()//运行游戏
{
while(1)
{
gotoxy(54,8);
printf("Your Score:%d",score);
/*
以下确保不能向上运动时改方向为向下等情况
*/
if(GetAsyncKeyState(VK_UP)&&ch!=S)
ch=W;
else if(GetAsyncKeyState(VK_DOWN)&&ch!=W)
ch=S;
else if(GetAsyncKeyState(VK_LEFT)&&ch!=D)
ch=A;
else if(GetAsyncKeyState(VK_RIGHT)&&ch!=A)
ch=D;
/*
根据分数和Sleep函数来确定游戏难度
*/
if(score>=50&&score<100)
sleep=300;
else if(score>=100&&score<300)
sleep=250;
else if(score>=300)
sleep=200;
if(GetAsyncKeyState(VK_SPACE))//输入space暂停游戏
{
while(1)
{
Sleep(300);
if(GetAsyncKeyState(VK_SPACE))
break;
}
}
else if (GetAsyncKeyState(VK_ESCAPE))//输入ESC直接结束游戏
{
game_flag=3;
gameover();
}
Sleep(sleep);
move();
}
}
/*
游戏结束界面
*/
void gameover()//游戏结束界面
{
FILE *fp;
system("cls");//清屏
gotoxy(48/2,26/2-2);
printf("\tGame Over!!!");//打印出游戏结束界面
gotoxy(48/2,26/2);
if(game_flag==1)
printf("\t你撞墙了!!!\n");
else if(game_flag==2)
printf("\t傻孩子!你不能吃你自己!!!\n");
else if(game_flag==3)
printf("\t您已结束游戏!");
gotoxy(48/2,26/2+2);
printf("\tYour score:%d\n",score);//打印出得到的分数
if(score>bestscore)//如果此次游戏分数大于以前最高分
{
fp=fopen("score.1","w");
fprintf(fp,"%d",score);//将此次分数保存在最高分文件里
fclose(fp);
}
system("pause");
exit(0);
}
/*
主函数
*/
int main()
{
system("color 9");
map_creat();
ini_snack();
cre_food();
rungame();
return 0;
}
更多有趣的经典小游戏实现专题,分享给大家:
C++经典小游戏汇总
python经典小游戏汇总
python俄罗斯方块游戏集合
JavaScript经典游戏 玩不停
javascript经典小游戏汇总
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
以上是 C语言实现链表贪吃蛇 的全部内容, 来源链接: utcz.com/p/245287.html