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;
}

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

(0)

相关推荐

  • 必须知道的C语言八大排序算法(收藏)

    概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的排序记录,在排序过程中需要访问外存. 我们这里说说八大排序就是内部排序. 当n较大,则应采用时间复杂度为O(nlog2n)的排序方法:快速排序.堆排序或归并排序序. 快速排序:是目前基于比较的内部排序中被认为是最好的方法,当待排序的关键字是随机分布时,快速排序的平均时间最短: 1.插入排序-直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到

  • C语言单链表实现多项式相加

    本文实例为大家分享了C语言单链表实现多项式相加的具体代码,供大家参考,具体内容如下 //多项式的相加和相乘 #include<stdio.h> #include<stdlib.h> #pragma warning(disable:4996)//兼容scanf typedef struct node { int coef; int expon; struct node* link; }Polynode,*Polynomial; Polynomial InsertPolyLinklis

  • C语言树状数组的实例详解

    C语言树状数组的实例详解 最近学了树状数组,给我的感觉就是 这个数据结构好神奇啊^_^ 首先她的常数比线段树小,其次她的实现复杂度也远低于线段树 (并没有黑线段树的意思=-=) 所以熟练掌握她是非常有必要的.. 关于树状数组的基础知识与原理网上一搜一大堆,我就不赘述了,就谈一些树状数组的应用好了 1,单点修改,求区间和 #define lowbit(x) (x&-x) // 设 x 的末尾零的个数为 y , 则 lowbit(x) == 2^y void Update(int i,int v)

  • C语言实现的学生选课系统代码分享

    好久没玩过C语言了,上一次还是在大二的时候...废话不多说,这里有一个C语言实现的学生选课系统代码,分享给大家,具体如下: #include<stdio.h> #include<stdlib.h> int N1,N2,kk1,kk2,kk3; struct couse * head1; struct student * head2; struct couse//课程信息结构体 { int num1; char name1[20]; int score; int nelepeo; /

  • linux下c语言的多线程编程

    我们在写linux的服务的时候,经常会用到linux的多线程技术以提高程序性能 多线程的一些小知识: 一个应用程序可以启动若干个线程. 线程(Lightweight Process,LWP),是程序执行的最小单元. 一般一个最简单的程序最少会有一个线程,就是程序本身,也就是主函数(单线程的进程可以简单的认为只有一个线程的进程) 一个线程阻塞并不会影响到另外一个线程. 多线程的进程可以尽可能的利用系统CPU资源. 1创建线程 先上一段在一个进程中创建一个线程的简单的代码,然后慢慢深入. #incl

  • 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 SNAK

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

    本文实例为大家分享了C语言链表实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 贪吃蛇设计思路: 屏幕坐标: 拓展功能: 1.F1,F2控制加速减速  空格暂停游戏  Esc退出 2.加速每个食物得分更高 先打印出游戏界面,还有初始化蛇,蛇的节点用字符串★表示,游戏背景用■表示,因为这些字符串占两个字节的宽度,所以每次x,y坐标的对应关系是x=y*2.在相应位置打印出蛇,初始化蛇为五个节点 初始化蛇头的移动方向为右,根据按键来确定蛇的移动状态,要是吃到食物,就让食物为蛇头,然后随机生成食物.

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

    本文实例为大家分享了C语言链表实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 项目名称: 贪吃蛇小游戏 运行环境: Linux 编程语言: C语言 主要语法: 链表,指针,函数 备注: 游戏中可选不同难度模式, 1.简易--Easy--速度慢,可穿墙,可触碰自己 2.困难--Hard--速度快,不可穿墙,不可触碰自己 3.自动--Auto--外挂模式,自动吃食,直到胜利 代码 贪吃蛇小游戏代码: #include <curses.h> #include <stdlib.h> #

  • 基于C语言实现的贪吃蛇游戏完整实例代码

    本文以实例的形式讲述了基于C语言实现的贪吃蛇游戏代码,这是一个比较常见的游戏,代码备有比较详细的注释,对于读者理解有一定的帮助. 贪吃蛇完整实现代码如下: #include <graphics.h> #include <conio.h> #include <stdlib.h> #include <dos.h> #define NULL 0 #define UP 18432 #define DOWN 20480 #define LEFT 19200 #defi

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

    本文实例为大家分享了C语言表实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 总体思想 利用循环链表将一条蛇的坐标进行存储,然后利用gotoxy()函数(可以将光标定位到指定的位置),此时根据蛇的坐标进行输出"@",输出多几个既可以产生一条蛇.通过遍历循环链表进行蛇的移动,对循环链表的插入元素,产生蛇变长的效果.下面为各功能实现的函数 1.贪吃蛇地图函数map() 2.蛇的移动move(),up(),left()等函数 3.产生食物food()和吃到食物eat_food() 4.蛇吃

  • C语言实现双人贪吃蛇游戏实例代码

    贪吃蛇双人小游戏,每局游戏两分钟,死亡则直接失败,若时间结束,则分高者获胜.   上源代码: ​ #include <stdio.h> #include <stdlib.h> #include <Windows.h> #include <time.h> #include<stdbool.h> #include <conio.h> #define SNAKESIZE 100 #define MAPWIDTH 118 #define MA

  • C语言开发实现贪吃蛇游戏

    本文实例为大家分享了C语言实现贪吃蛇游戏的具体代码,供大家参考,具体内容如下 1.最好用VS运行 2.用到的函数有:_kbhit _getch EasyX图形库内一系列函数 3.蛇身与食物用矩形画的 代码如下: #include<stdio.h> #include<graphics.h> //easyx头文件 #include<conio.h> #include<time.h> /********************设置各种属性*************

  • unity使用链表实现贪吃蛇游戏

    今天介绍一下如何利用链表结构来创建一条贪吃蛇. 要实现的功能很简单,按下空格键使蛇加长一节,每次按下空格就在蛇尾加一个cube.按下左方向键,控制蛇的移动.如图所示: //贪吃蛇的中心是:定义蛇身第一节,即链表头为temp.链表的子节点为next.蛇头带着temp(链表头)走,temp带着next(链表子节点)走. 1. 创建一个cube设为预设体,作为蛇身的节点.在预设体上面添加BodyScript脚本. using UnityEngine; using System.Collections;

  • C语言实现贪吃蛇游戏

    最近整理下电脑,看到了自己在上个学期打的贪吃蛇游戏的c代码,觉得真的是略微有点冗长,但是实现起来应该也算是比较好理解,于是把自己的代码搬上来,网络上写贪吃蛇的c语言的文章很多,我这篇也仅是给大家作为一个参考而已. 我的代码是在Windows下运行的,因为需要用到windows.h这个库. 然后也做了一个简单的ai模式,这在没有障碍物的情况下前期还是蛮不错的,但是到了后期蛇变长了之后就会有bug了. 好了,直接上代码吧: 1)头文件和宏定义 #include<stdio.h> #include&

  • 70行C语言代码实现贪吃蛇

    本文实例为大家分享了C语言实现贪吃蛇的具体代码,供大家参考,具体内容如下 #include <stdio.h> #include <Windows.h> #include <conio.h> #include <time.h> #define MAX_WIDE 50 #define MAX_HIGH 16 short dx = 1, dy = 0, randxy, score = 0; COORD coord; struct Snake{ short len

随机推荐