当前位置: 首页 > 编程笔记 >

C语言链表实现贪吃蛇游戏

尉迟鸿熙
2023-03-14
本文向大家介绍C语言链表实现贪吃蛇游戏,包括了C语言链表实现贪吃蛇游戏的使用技巧和注意事项,需要的朋友参考一下

阅读学习了源代码,并做了简单的注释和修改,里面只用了链表数据结构,非常适合C语言入门者学习阅读。

程序可在VS2013下编译运行。

#include<stdio.h>
#include<time.h>
#include<windows.h>
#include<stdlib.h>

#define U 1
#define D 2
#define L 3 
#define R 4  //蛇的状态,U:上 ;D:下;L:左 R:右

typedef struct SNAKE //蛇身的一个节点
{
 int x;
 int y;
 struct SNAKE *next;
}snake;

//全局变量//
int score = 0, add = 10;//总得分与每次吃食物得分。
int status, sleeptime = 200;//每次运行的时间间隔
snake *head, *food;//蛇头指针,食物指针
snake *q;//遍历蛇的时候用到的指针
int endGamestatus = 0; //游戏结束的情况,1:撞到墙;2:咬到自己;3:主动退出游戏。

//声明全部函数//
void Pos();
void creatMap();
void initSnake();
int biteSelf();
void createFood();
void cantCrossWall();
void snakeMove();
void pause();
void runGame();
void initGame();
void endGame();
void gameStart();

void Pos(int x, int y)//设置光标位置
{
 COORD pos;
 HANDLE hOutput;
 pos.X = x;
 pos.Y = y;
 hOutput = GetStdHandle(STD_OUTPUT_HANDLE);//返回标准的输入、输出或错误的设备的句柄,也就是获得输入、输出/错误的屏幕缓冲区的句柄
 SetConsoleCursorPosition(hOutput, pos);
}

void creatMap()//创建地图
{
 int i;
 for (i = 0; i<58; i += 2)//打印上下边框
 {
  Pos(i, 0);
  printf("■");//一个方块占两个位置
  Pos(i, 26);
  printf("■");
 }
 for (i = 1; i<26; i++)//打印左右边框
 {
  Pos(0, i);
  printf("■");
  Pos(56, i);
  printf("■");
 }
}

void initSnake()//初始化蛇身
{
 snake *tail;
 int i;
 tail = (snake*)malloc(sizeof(snake));//从蛇尾开始,头插法,以x,y设定开始的位置//
 tail->x = 24;
 tail->y = 5;
 tail->next = NULL;
 for (i = 1; i <= 4; i++)//初始长度为4
 {
  head = (snake*)malloc(sizeof(snake));
  head->next = tail;
  head->x = 24 + 2 * i;
  head->y = 5;
  tail = head;
 }
 while (tail != NULL)//从头到为,输出蛇身
 {
  Pos(tail->x, tail->y);
  printf("■");
  tail = tail->next;
 }
}
//??
int biteSelf()//判断是否咬到了自己
{
 snake *self;
 self = head->next;
 while (self != NULL)
 {
  if (self->x == head->x && self->y == head->y)
  {
   return 1;
  } 
  self = self->next;
 }
 return 0;
}

void createFood()//随机出现食物
{
 snake *food_1;
 srand((unsigned)time(NULL));//为了防止每次产生的随机数相同,种子设置为time
 food_1 = (snake*)malloc(sizeof(snake));
 while ((food_1->x % 2) != 0) //保证其为偶数,使得食物能与蛇头对其
 {
  food_1->x = rand() % 52 + 2;
 }
 food_1->y = rand() % 24 + 1;
 q = head;
 while (q->next == NULL)
 {
  if (q->x == food_1->x && q->y == food_1->y) //判断蛇身是否与食物重合
  {
   free(food_1);
   createFood();
  }
  q = q->next;
 }
 Pos(food_1->x, food_1->y);
 food = food_1;
 printf("■");
}

void cantCrossWall()//不能穿墙
{
 if (head->x == 0 || head->x == 56 || head->y == 0 || head->y == 26)
 {
  endGamestatus = 1;
  endGame();
 }
}

void snakeMove()//蛇前进,上U,下D,左L,右R
{
 snake * nexthead;
 cantCrossWall();

 nexthead = (snake*)malloc(sizeof(snake));
 if (status == U)
 {
  nexthead->x = head->x;
  nexthead->y = head->y - 1;
  if (nexthead->x == food->x && nexthead->y == food->y)//如果下一个有食物//
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   score = score + add;
   createFood();
  }
  else            //如果没有食物//
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q->next->next != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   Pos(q->next->x, q->next->y);
   printf(" ");
   free(q->next);
   q->next = NULL;
  }
 }
 if (status == D)
 {
  nexthead->x = head->x;
  nexthead->y = head->y + 1;
  if (nexthead->x == food->x && nexthead->y == food->y) //有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   score = score + add;
   createFood();
  }
  else        //没有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q->next->next != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   Pos(q->next->x, q->next->y);
   printf(" ");
   free(q->next);
   q->next = NULL;
  }
 }
 if (status == L)
 {
  nexthead->x = head->x - 2;
  nexthead->y = head->y;
  if (nexthead->x == food->x && nexthead->y == food->y)//有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   score = score + add;
   createFood();
  }
  else        //没有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q->next->next != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   Pos(q->next->x, q->next->y);
   printf(" ");
   free(q->next);
   q->next = NULL;
  }
 }
 if (status == R)
 {
  nexthead->x = head->x + 2;
  nexthead->y = head->y;
  if (nexthead->x == food->x && nexthead->y == food->y)//有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   score = score + add;
   createFood();
  }
  else           //没有食物
  {
   nexthead->next = head;
   head = nexthead;
   q = head;
   while (q->next->next != NULL)
   {
    Pos(q->x, q->y);
    printf("■");
    q = q->next;
   }
   Pos(q->next->x, q->next->y);
   printf(" ");
   free(q->next);
   q->next = NULL;
  }
 }
 if (biteSelf() == 1)  //判断是否会咬到自己
 {
  endGamestatus = 2;
  endGame();
 }
}

void pause()//暂停
{
 while (1)
 {
  Sleep(300);
  if (GetAsyncKeyState(VK_SPACE))
  {
   break;
  }

 }
}

void runGame()//控制游戏  
{

 Pos(64, 15);
 printf("不能穿墙,不能咬到自己\n");
 Pos(64, 16);
 printf("用↑.↓.←.→分别控制蛇的移动.");
 Pos(64, 17);
 printf("F1 为加速,F2 为减速\n");
 Pos(64, 18);
 printf("ESC :退出游戏.space:暂停游戏.");
 Pos(64, 20);
 printf("C语言研究中心 www.clang.cc");
 status = R;
 while (1)
 {
  Pos(64, 10);
  printf("得分:%d ", score);
  Pos(64, 11);
  printf("每个食物得分:%d分", add);
  if (GetAsyncKeyState(VK_UP) && status != D)
  {
   status = U;
  }
  else if (GetAsyncKeyState(VK_DOWN) && status != U)
  {
   status = D;
  }
  else if (GetAsyncKeyState(VK_LEFT) && status != R)
  {
   status = L;
  }
  else if (GetAsyncKeyState(VK_RIGHT) && status != L)
  {
   status = R;
  }
  else if (GetAsyncKeyState(VK_SPACE))
  {
   pause();
  }
  else if (GetAsyncKeyState(VK_ESCAPE))
  {
   endGamestatus = 3;
   break;
  }
  else if (GetAsyncKeyState(VK_F1))
  {
   if (sleeptime >= 50)
   {
    sleeptime = sleeptime - 30;
    add = add + 2;
    if (sleeptime == 320)
    {
     add = 2;//防止减到1之后再加回来有错
    }
   }
  }
  else if (GetAsyncKeyState(VK_F2))
  {
   if (sleeptime<350)
   {
    sleeptime = sleeptime + 30;
    add = add - 2;
    if (sleeptime == 350)
    {
     add = 1; //保证最低分为1
    }
   }
  }
  Sleep(sleeptime);
  snakeMove();
 }
}

void initGame()//开始界面
{
 Pos(40, 12);

 system("title C语言研究中心 www.clang.cc");
 printf("欢迎来到贪食蛇游戏!");
 Pos(40, 25);
 printf("    C语言研究中心 www.clang.cc.\n");
 system("pause");
 system("cls");
 Pos(25, 12);
 printf("用↑.↓.←.→分别控制蛇的移动, F1 为加速,2 为减速\n");
 Pos(25, 13);
 printf("加速将能得到更高的分数。\n");
 system("pause");
 system("cls");
}

void endGame()//结束游戏
{

 system("cls");
 Pos(24, 12);
 if (endGamestatus == 1)
 {
  printf("对不起,您撞到墙了。游戏结束.");
 }
 else if (endGamestatus == 2)
 {
  printf("对不起,您咬到自己了。游戏结束.");
 }
 else if (endGamestatus == 3)
 {
  printf("您的已经结束了游戏。");
 }
 Pos(24, 13);
 printf("您的得分是%d\n", score);
 while (getchar() != 'y')
 { 
  printf("close?[y]");
 }
 exit(0);
}

void gameStart()//游戏初始化
{
 system("mode con cols=100 lines=30");
 initGame();
 creatMap();
 initSnake();
 createFood();
}

int main()
{
 gameStart();
 runGame();
 endGame();
 return 0;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。

 类似资料:
  • 本文向大家介绍C语言实现链表贪吃蛇,包括了C语言实现链表贪吃蛇的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家分享了C语言实现贪吃蛇的具体代码,供大家参考,具体内容如下 用C语言链表写的贪吃蛇(程序设计时做的,做的不好大佬勿喷) 借助游戏内容分析贪吃蛇所需的功能主要包括这几块: 1.移动光标模块 2.打印地图模块和基本规则信息 读取最高分文件 3.打印初始蛇模块 打印时给予蛇的初始移动方向

  • 本文向大家介绍C语言循环链表实现贪吃蛇游戏,包括了C语言循环链表实现贪吃蛇游戏的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家分享了C语言表实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 总体思想 利用循环链表将一条蛇的坐标进行存储,然后利用gotoxy()函数(可以将光标定位到指定的位置),此时根据蛇的坐标进行输出“@”,输出多几个既可以产生一条蛇。通过遍历循环链表进行蛇的移动,对循

  • 本文向大家介绍C语言单链表贪吃蛇小游戏,包括了C语言单链表贪吃蛇小游戏的使用技巧和注意事项,需要的朋友参考一下 C语言实现单链表控制台贪吃蛇小游戏,供大家参考。 编译环境:vs2019 需求: 统计游戏开始后的时间,控制贪吃蛇;吃到食物蛇身加长,得分加一;碰墙或蛇头碰到身体减一条生命;生命消耗完则结束游戏。 思路: 使用wasd键控制蛇的移动方向,蛇头碰到食物得分加一,并在地图上随机产生一个食物,

  • 本文向大家介绍C语言贪吃蛇经典小游戏,包括了C语言贪吃蛇经典小游戏的使用技巧和注意事项,需要的朋友参考一下 一、贪吃蛇小游戏简介: 用上下左右控制蛇的方向,寻找吃的东西,每吃一口就能得到一定的积分,而且蛇的身子会越吃越长,身子越长玩的难度就越大,不能碰墙,也不能咬到自己的身体,等到了一定的分数,就能过关。 二、函数框架 三、数据结构 定义蛇的结构体,利用单链表来表示蛇,每个结点为蛇身体的一部分。

  • 本文向大家介绍C#贪吃蛇游戏实现分析,包括了C#贪吃蛇游戏实现分析的使用技巧和注意事项,需要的朋友参考一下 今天无聊突发奇想做个贪吃蛇,虽然网上很多这东西了,不过自己写的感觉还行吧 贪吃蛇分析 游戏规则: 1、蛇起始长度5,每吃一个食物增加1,最大15过关 2、蛇用蓝色表示,食物用绿色,障碍物用黑色 3、当蛇碰到自己、墙壁、障碍物则游戏失败 4、方向键控制蛇的移动方向,蛇不可反方向移动,如正在向上

  • 本文向大家介绍python实现贪吃蛇游戏,包括了python实现贪吃蛇游戏的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家分享了python实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 本文稍作改动,修复一些bug,原文链接:python实现贪吃蛇游戏 以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持呐喊教程。