c语言贪吃蛇详解4.食物的投放与蛇的变长

前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解。我将分几步来教大家写一个贪吃蛇小游戏。由于大家c语言未学完,这个教程只涉及数组和函数等知识点。

通过前几次的教程,我们已经做出来了能上下左右跑的小蛇了。现在我们就先来做下食物投放吧。

食物投放的基本思想是在地图上随机找一个没有蛇身也没有障碍物的地方,然后把这个地方的地图数组值标记为-1(前面我们让空地为0,障碍物为1)。

我们先来写一个函数来判断一个点是不是符合上面的条件。

int check(int ii,int jj)        //判断这个点能不能放食物,可以放返回1,不能放返回0
{
if(a[ii][jj]==) //如果有障碍物,返回0
return ;
int i;
for(i=; i<sLength; i++)
{
if(ii==s[i]&&jj==s[i][]) //如果和其中一个蛇身重合,就返回0
return ;
}
if(ii==||ii==H-||jj==||jj==W-) //如果在边界上面,返回0
return ;
return ; //最后筛选 过后的是符合条件的点
}

要使用随机数,先

#include <stdlib.h>

然后由于要根据时间设置随机数种子,所以要

#include<time.h>

在init函数里面写这句代码

srand((unsigned)time(NULL));            //设置随机数种子为现在的时间

然后写一个food函数,实现投放一个食物

void food()
{
int i,j;
do
{
i=rand()%H; //生成0~H-1之间的一个数
j=rand()%W;
}while(check(i,j)==); //生成点直到满足条件
a[i][j]=-; //标记为食物
gotoxy(i,j);
printf("$"); //画出食物
}

然后在main里面的循环前面调用一次food,实现开始游戏时有一个食物。然后检查蛇头与食物是否重合,如果重合,就调用一次food,投放一个食物。

int main()
{
init(); //程序开始时的初始化操作
drawMap(); //画地图
food();
while()
{
drawSnake(); //画蛇
Sleep(WAIT_TIME); //等待一段时间
key();
move(); //移动蛇(主要是修改蛇身数组的数据)
if(a[s[][]][s[][]]==-) //如果蛇头碰到食物,就重新投放食物,并且把食物点重置为0
{
food();
a[s[][]][s[][]]=;
} }
getchar();
return ;
}

让我们来看看效果

到现在为止的代码:

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include<time.h>
#include<windows.h> #define H 23
#define W 75
#define WAIT_TIME 500 //定义direction的每个值代表的方向
#define UP 0
#define DOWN 1
#define LEFT 2
#define RIGHT 3 int a[H][W]; //地图数组
int s[H*W][]; //蛇身坐标数组
int sLength; //蛇的长度
int direction; //蛇的方向 void init() //程序开始时的初始化操作
{
srand((unsigned)time(NULL)); //设置随机数种子为现在的时间
CONSOLE_CURSOR_INFO cursor_info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cursor_info); //隐藏关标
int i,j;
for(i=; i<H; i++)
{
a[i][]=; //让第一列为1
a[i][W-]=; //让最后一列为1
}
for(j=; j<W; j++)
{
a[][j]=; //让第一行为1
a[H-][j]=; //让最后一行为1
}
sLength=; //让蛇的最初长度为4
s[][]=H/;
s[][]=W/; //给蛇头坐标赋值
for(i=; i<; i++)
{
s[i][]=s[][]+i;
s[i][]=s[][]; //给刚开始的蛇身几个初始坐标
}
direction=UP;
} void gotoxy(int i,int j) //移动光标
{
COORD position= {j,i};
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE),position);
} int check(int ii,int jj) //判断这个点能不能放食物,可以放返回1,不能放返回0
{
if(a[ii][jj]==) //如果有障碍物,返回0
return ;
int i;
for(i=; i<sLength; i++)
{
if(ii==s[i][]&&jj==s[i][]) //如果和其中一个蛇身重合,就返回0
return ;
}
if(ii==||ii==H-||jj==||jj==W-) //如果在边界上面,返回0
return ;
return ; //最后筛选 过后的是符合条件的点
} void food()
{
int i,j;
do
{
i=rand()%H; //生成0~H-1之间的一个数
j=rand()%W;
}
while(check(i,j)==); //生成点直到满足条件
a[i][j]=-; //标记为食物
gotoxy(i,j);
printf("$"); //画出食物
} void drawMap() //画地图
{
gotoxy(,);
int i,j;
for(i=; i<H; i++)
{
for(j=; j<W; j++) //两重for循环遍历数组
{
if(a[i][j]==) //为0输出空格
printf(" ");
else //为1输出#
printf("#");
}
printf("\n"); //别忘了换行
}
} void move()
{
int i;
gotoxy(s[sLength-][],s[sLength-][]);
printf(" "); //在尾巴上面画空格以擦除尾巴
for(i=sLength-; i>; i--) //从尾巴开始,每一个点的位置等于它前面一个点的位置
{
s[i][]=s[i-][];
s[i][]=s[i-][];
}
switch(direction)
{
case UP:
s[][]--;
break;
case DOWN:
s[][]++;
break;
case LEFT:
s[][]--;
break;
case RIGHT:
s[][]++;
break;
} } void drawSnake() //画蛇
{
int i;
for(i=; i<sLength; i++)
{
gotoxy(s[i][],s[i][]); //移动关标到蛇的坐标
printf("@"); //在这个位置画蛇
}
} void key()
{
if(kbhit()!=) //如果有键盘输入
{
char in;
while(!kbhit()==) //如果玩家输入了多个按键,以最后一个按键为准
in=getch();
switch(in)
{
case 'w':
case 'W':
if(direction!=DOWN) //不能缩头吧。。。。
direction=UP;
break;
case 's':
case 'S':
if(direction!=UP)
direction=DOWN;
break;
case 'a':
case 'A':
if(direction!=RIGHT)
direction=LEFT;
break;
case 'd':
case 'D':
if(direction!=LEFT)
direction=RIGHT;
break;
}
}
} int main()
{
init(); //程序开始时的初始化操作
drawMap(); //画地图
food();
while()
{
drawSnake(); //画蛇
Sleep(WAIT_TIME); //等待一段时间
key();
move(); //移动蛇(主要是修改蛇身数组的数据)
if(a[s[][]][s[][]]==-) //如果蛇头碰到食物,就重新投放食物,并且把食物点重置为0
{
food();
a[s[][]][s[][]]=;
} }
getchar();
return ;
}

好了,现在食物能正常投放了

接下来我们就来实现一下蛇吃到食物的变长功能吧。

首先设置一个变量标记蛇是否吃到食物

bool eated=false;   //标记是否吃到食物

然后在main里面的判断吃到食物时,使eated变为true

if(a[s[][]][s[][]]==-) //如果蛇头碰到食物,就重新投放食物,并且把食物点重置为0
{
eated=true; //标记已经吃到食物
food();
a[s[][]][s[][]]=; //去掉食物
}

然后在move函数里面加下面的代码

if(eated)               //如果吃到了食物
{
sLength++;
eated=false; //设置为false,不然无限变长
}

这样在下面的坐标移动环节,就不会舍弃掉上一次的尾巴节点。蛇就变长了。

来看看效果:

蛇可以变长了。不过有时候玩着玩着有事,又舍不得玩了这么长的蛇,要是有个暂停功能就好了。接下来我们就来顺便实现下。

在key函数里面的switch里面加一个case

case 'p':
case 'P':
gotoxy(H,); //将光标移动到下面
system("pause");
gotoxy(H,);
printf(" "); //消去下面的按任意键继续
break;

这样就可以实现按p键暂停了。(记得玩的时候切换为英文输入法呀)

来看看到现在为止的代码:

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include<time.h>
#include<windows.h> #define H 23
#define W 75
#define WAIT_TIME 500 //定义direction的每个值代表的方向
#define UP 0
#define DOWN 1
#define LEFT 2
#define RIGHT 3 int a[H][W]; //地图数组
int s[H*W][]; //蛇身坐标数组
int sLength; //蛇的长度
int direction; //蛇的方向
bool eated=false; //标记是否吃到食物 void init() //程序开始时的初始化操作
{
srand((unsigned)time(NULL)); //设置随机数种子为现在的时间
CONSOLE_CURSOR_INFO cursor_info = {, };
SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), &cursor_info); //隐藏关标
int i,j;
for(i=; i<H; i++)
{
a[i][]=; //让第一列为1
a[i][W-]=; //让最后一列为1
}
for(j=; j<W; j++)
{
a[][j]=; //让第一行为1
a[H-][j]=; //让最后一行为1
}
sLength=; //让蛇的最初长度为4
s[][]=H/;
s[][]=W/; //给蛇头坐标赋值
for(i=; i<; i++)
{
s[i][]=s[][]+i;
s[i][]=s[][]; //给刚开始的蛇身几个初始坐标
}
direction=UP;
} void gotoxy(int i,int j) //移动光标
{
COORD position= {j,i};
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE),position);
} int check(int ii,int jj) //判断这个点能不能放食物,可以放返回1,不能放返回0
{
if(a[ii][jj]==) //如果有障碍物,返回0
return ;
int i;
for(i=; i<sLength; i++)
{
if(ii==s[i][]&&jj==s[i][]) //如果和其中一个蛇身重合,就返回0
return ;
}
if(ii==||ii==H-||jj==||jj==W-) //如果在边界上面,返回0
return ;
return ; //最后筛选 过后的是符合条件的点
} void food()
{
int i,j;
do
{
i=rand()%H; //生成0~H-1之间的一个数
j=rand()%W;
}
while(check(i,j)==); //生成点直到满足条件
a[i][j]=-; //标记为食物
gotoxy(i,j);
printf("$"); //画出食物
} void drawMap() //画地图
{
gotoxy(,);
int i,j;
for(i=; i<H; i++)
{
for(j=; j<W; j++) //两重for循环遍历数组
{
if(a[i][j]==) //为0输出空格
printf(" ");
else //为1输出#
printf("#");
}
printf("\n"); //别忘了换行
}
} void move()
{
int i;
gotoxy(s[sLength-][],s[sLength-][]);
printf(" "); //在尾巴上面画空格以擦除尾巴
if(eated) //如果吃到了食物
{
sLength++;
eated=false; //设置为false,不然无限变长
}
for(i=sLength-; i>; i--) //从尾巴开始,每一个点的位置等于它前面一个点的位置
{
s[i][]=s[i-][];
s[i][]=s[i-][];
}
switch(direction)
{
case UP:
s[][]--;
break;
case DOWN:
s[][]++;
break;
case LEFT:
s[][]--;
break;
case RIGHT:
s[][]++;
break;
} } void drawSnake() //画蛇
{
int i;
for(i=; i<sLength; i++)
{
gotoxy(s[i][],s[i][]); //移动关标到蛇的坐标
printf("@"); //在这个位置画蛇
}
} void key()
{
if(kbhit()!=) //如果有键盘输入
{
char in;
while(!kbhit()==) //如果玩家输入了多个按键,以最后一个按键为准
in=getch();
switch(in)
{
case 'w':
case 'W':
if(direction!=DOWN) //不能缩头吧。。。。
direction=UP;
break;
case 's':
case 'S':
if(direction!=UP)
direction=DOWN;
break;
case 'a':
case 'A':
if(direction!=RIGHT)
direction=LEFT;
break;
case 'd':
case 'D':
if(direction!=LEFT)
direction=RIGHT;
break;
case 'p':
case 'P':
gotoxy(H,); //将光标移动到下面
system("pause");
gotoxy(H,);
printf(" "); //消去下面的按任意键继续
break;
}
}
} int main()
{
init(); //程序开始时的初始化操作
drawMap(); //画地图
food();
while()
{
drawSnake(); //画蛇
Sleep(WAIT_TIME); //等待一段时间
key();
move(); //移动蛇(主要是修改蛇身数组的数据)
if(a[s[][]][s[][]]==-) //如果蛇头碰到食物,就重新投放食物,并且把食物点重置为0
{
eated=true; //标记已经吃到食物
food();
a[s[][]][s[][]]=; //去掉食物
} }
getchar();
return ;
}

c语言贪吃蛇详解4.食物的投放与蛇的变长的更多相关文章

  1. c语言贪吃蛇详解5.GameOver功能与显示成绩

    c语言贪吃蛇详解5.GameOver功能与显示成绩 以前我们已经做出来了一个能吃东西变长的蛇.不过它好像不会死... 现在就来实现一下game over的功能吧. 写个函数判断蛇是否撞到自己或者撞到墙 ...

  2. c语言贪吃蛇详解3.让蛇动起来

    c语言贪吃蛇详解3.让蛇动起来 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 上次 ...

  3. c语言贪吃蛇详解-2.画出蛇

    c语言贪吃蛇详解-2.画出蛇 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 蛇的身 ...

  4. c语言贪吃蛇详解1.画出地图

    c语言贪吃蛇详解-1.画出地图 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 首先 ...

  5. C语言内存对齐详解(2)

    接上一篇:C语言内存对齐详解(1) VC对结构的存储的特殊处理确实提高CPU存储变量的速度,但是有时候也带来了一些麻烦,我们也屏蔽掉变量默认的对齐方式,自己可以设定变量的对齐方式.VC 中提供了#pr ...

  6. C语言内存对齐详解(3)

    接上一篇:C语言内存对齐详解(2) 在minix的stdarg.h文件中,定义了如下一个宏: /* Amount of space required in an argument list for a ...

  7. 一个简单的C语言程序(详解)

    C Primer Plus之一个简单的C语言程序(详解) #include <stdio.h> int main(void) //一个简单的 C程序 { int num; //定义一个名为 ...

  8. [转帖]rename(Perl语言版本) 详解

    rename(Perl语言版本) 详解 2019-03-19 22:51:23 wayne17 阅读数 464更多 分类专栏: Ubuntu之路   版权声明:本文为博主原创文章,遵循CC 4.0 B ...

  9. C语言memset函数详解

    C语言memset函数详解 memset() 的作用:在一段内存块中填充某个给定的值,通常用于数组初始化与数组清零. 它是直接操作内存空间,mem即“内存”(memory)的意思.该函数的原型为: # ...

随机推荐

  1. 微信小程序去除button按钮的边框

    小程序开发记录 小程序开发中, 有时候我们希望button不要有边框, 需要使用button::after来实现, 具体如下: .operations button::after{border:0 n ...

  2. 【剑指offer】扑克牌的顺子

    个大王,2个小王(一副牌原本是54张^_^)...他随机从中抽出了5张牌,想測測自己的手气,看看能不能抽到顺子,假设抽到的话,他决定去买体育彩票,嘿嘿! ."红心A,黑桃3,小王,大王,方片 ...

  3. scrapy爬虫框架setting模块解析

    平时写爬虫的时候并不需要设置setting里所有的参数,今天心血来潮,花了点时间查了一下setting模块创建后自动写入的所有参数的含义,记录一下. 模块相关说明信息 # -*- coding: ut ...

  4. (转自知乎https://www.zhihu.com/question/20794107)动态代理

    作者:雨夜偷牛的人链接:https://www.zhihu.com/question/20794107/answer/23330381来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载 ...

  5. spring mvc获取绝对路径的几种方法

    1.首先如果是在一个controller方法中,则很简单,直接用下面语句. @RequestMapping("categoryHome") public ModelAndView ...

  6. MQTT 简介

    MQTT 全称是 Message Queue Telemetry Transport,是一个轻量级的“发布/订阅”消息传输协议. 官网 http://mqtt.org/ 发布/订阅 MQTT 的基本概 ...

  7. Git使用简单总结

    创建版本库git add加入到暂存区git commit -m" "加入到分支 时光机穿梭git satus查看仓库的当前状态git diff file 查看修改内容 版本回退HE ...

  8. spring-boot学习笔记之Conditional

    今天看了@Conditional,自己根据以下文章练了下,根据自己的理解操作的             转载出处:http://wiselyman.iteye.com/blog/2213054 17. ...

  9. 《TCP-IP详解卷2:实现》【PDF】下载

    <TCP-IP详解卷2:实现>[PDF]下载链接: https://u253469.pipipan.com/fs/253469-230062539 内容简介 <TCP/IP详解·卷2 ...

  10. 【java】java.util.Arrays类常用方法

    package Arrays类; import java.util.Arrays; public class TestArrays { public static void main(String[] ...