c语言贪吃蛇详解4.食物的投放与蛇的变长
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.食物的投放与蛇的变长的更多相关文章
- c语言贪吃蛇详解5.GameOver功能与显示成绩
c语言贪吃蛇详解5.GameOver功能与显示成绩 以前我们已经做出来了一个能吃东西变长的蛇.不过它好像不会死... 现在就来实现一下game over的功能吧. 写个函数判断蛇是否撞到自己或者撞到墙 ...
- c语言贪吃蛇详解3.让蛇动起来
c语言贪吃蛇详解3.让蛇动起来 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 上次 ...
- c语言贪吃蛇详解-2.画出蛇
c语言贪吃蛇详解-2.画出蛇 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 蛇的身 ...
- c语言贪吃蛇详解1.画出地图
c语言贪吃蛇详解-1.画出地图 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 首先 ...
- C语言内存对齐详解(2)
接上一篇:C语言内存对齐详解(1) VC对结构的存储的特殊处理确实提高CPU存储变量的速度,但是有时候也带来了一些麻烦,我们也屏蔽掉变量默认的对齐方式,自己可以设定变量的对齐方式.VC 中提供了#pr ...
- C语言内存对齐详解(3)
接上一篇:C语言内存对齐详解(2) 在minix的stdarg.h文件中,定义了如下一个宏: /* Amount of space required in an argument list for a ...
- 一个简单的C语言程序(详解)
C Primer Plus之一个简单的C语言程序(详解) #include <stdio.h> int main(void) //一个简单的 C程序 { int num; //定义一个名为 ...
- [转帖]rename(Perl语言版本) 详解
rename(Perl语言版本) 详解 2019-03-19 22:51:23 wayne17 阅读数 464更多 分类专栏: Ubuntu之路 版权声明:本文为博主原创文章,遵循CC 4.0 B ...
- C语言memset函数详解
C语言memset函数详解 memset() 的作用:在一段内存块中填充某个给定的值,通常用于数组初始化与数组清零. 它是直接操作内存空间,mem即“内存”(memory)的意思.该函数的原型为: # ...
随机推荐
- 一句话搞定webmap(一)——轻地图组件
摘要: 遥想当年.在APP中增加LBS元素相当困难:要刻苦学习java,要刻苦学习iOS开发,要刻苦学习javascript-- 而现在.要制作一张地图真是越来越easy了!居然仅仅须要一句话.就能够 ...
- Oracle 11gR2 用exp无法导出空表解决方法
Oracle 11gR2 用exp无法导出空表解决方法 在11gR2中有个新特性,当表无数据时,不分配segment以节省空间.Oracle 当然在运行export导出时,空表则无法导出,可是还是有解 ...
- ArcGIS教程:创建特征
摘要 创建由输入样本数据和一组栅格波段定义的类的 ASCII 特征文件. 使用方法 · 输出特征文件应使用扩展名 .gsg. · 输入栅格波段和输入栅格或要素样本数据必须具有重叠范围.将仅为公共区域计 ...
- shell 参数记录
$0 执行脚本的名称 $* 和 $@ 所有参数 $# 参数个数 $_ 上个命令的最后一个参数 $$ 代表当前命令所在的pid $! 代表最后执行的 '后台' 命令的pid $? 代表上条命令 ...
- Linux下查看Go语言软件运行情况
在Linux下,使用"jps"可以查看用Java语言写的软件的运行情况,如果要查看GO语言写的软件的运行情况,可以使用"gops",但这不是系统自带的,需要进行 ...
- 《调试九法——软硬件错误的排查之道》【PDF】下载
<调试九法--软硬件错误的排查之道>[PDF]下载链接: https://u253469.ctfile.com/fs/253469-231196352 内容简介 <调试九法:软硬件错 ...
- npm发布vue组件流程
初始化项目vue init webpack-simple XXX 定义组件略 发布配置1.package.json 2.webpack.config.js(注释部分为原配置) 发布1.登录 2.发布n ...
- Spring之AOP二
在Spring之AOP一中使用动态代理将日志打印功能注入到目标对象中,其实这就是AOP实现的原理,不过上面只是Java的实现方式.AOP不管什么语言它的几个主要概念还是有必要了解一下的. 一.AOP概 ...
- http性能测试工具wrk源码学习之开篇
1.前言 最近工作需要测试nginx反向代理的性能,于是找了一些http测试工具,例如经典的Apache的ab.siege.wrk.wrk使用多线程事件驱动方式,支持lua脚本扩展.关于wrk介绍可以 ...
- Java 哲学家进餐
某次操作系统实验存档.V 这个哲学家除了吃就知道睡.( ╯□╰ ) 哲学家.java: package operating.entity.philosophyeating; import operat ...