c语言贪吃蛇详解3.让蛇动起来
c语言贪吃蛇详解3.让蛇动起来
前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解。我将分几步来教大家写一个贪吃蛇小游戏。由于大家c语言未学完,这个教程只涉及数组和函数等知识点。
上次我已经教大家画出蛇了,现在我就教大家让蛇动起来。为了让大家更好理解,蛇的移动就用最简单的办法,这里就不用链表,顺序队列什么的了。蛇的移动就是头往前移动,然后后面的身体移动到前一节身体原来的位置上。
就像这样:
int i;
for(i=sLength-;i<;i--) //从尾巴开始,每一个点的位置等于它前面一个点的位置
{
s[i][]=s[i-][];
s[i][]=s[i-][];
}
那蛇头怎么办呢?
为了确定蛇头的位置,我们定义一个变量原来存储蛇现在的方向。
int direction; //蛇的方向
我们可以通过direction的值来标记现在蛇的方向。比如0代表上,1代表下。。。
为了方便,我们定义一些常量
//定义direction的每个值代表的方向
#define UP 0
#define DOWN 1
#define LEFT 2
#define RIGHT 3
这样,我们就可以这样写
switch(direction)
{
case UP:
s[][]--;
break;
case DOWN:
s[][]++;
break;
case LEFT:
s[][]--;
break;
case RIGHT:
s[][]++;
break;
}
写好的move函数如下
void move()
{
int i;
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;
} }
然后move写好了,我们还要写个循环,这样就可以让蛇一直动了。地图画完一次并且画了蛇之后,我们要把光标移动到起始位置。最简单的办法是每一次画地图的时候都把关标移动到(0,0)。
修改drawMap函数为下面这样。在开头加一个设置位置的语句。
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"); //别忘了换行
}
}
然后建立一个循环,不断地画蛇,移动。
int main()
{
init(); //程序开始时的初始化操作
drawMap(); //画地图
while()
{
drawSnake(); //画蛇
Sleep(WAIT_TIME); //等待一段时间
move(); //移动蛇(主要是修改蛇身数组的数据)
}
getchar();
return ;
}
然后观察一下,蛇确实走了,但是蛇的痕迹还在。

这样我们可以在蛇移动函数里面,擦去上一次的尾巴。
修改过后的move函数如下
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;
} }
运行一下,蛇动起来了。

然后试试修改下方向。在init函数最后写一个
direction=LEFT;
再运行试试

好,感觉很正常。
现在我们可以通过给direction变量赋值来改变蛇的前进方向了。
接下来,我们试试用键盘控制它。
写一个新函数key,用来处理键盘输入。
我们的输入不能在屏幕上有痕迹,并且不能输入一个按键就回车一下。。。所以我们用getch函数。
加conio.h头文件。然后getch会暂停程序等待输入,而蛇只在转弯的时候需要输入。所以我们就需要一个判断有没有输入的函数

然后通过输入的字符是w还是a还是s或者d来控制蛇的方向。
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;
}
}
}
好了,现在我们就能通过wasd这四个按键控制蛇到处走了。(进入游戏先切换为英文输入法)

下面附这一步的完整代码。
#include <stdio.h>
#include <stdlib.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() //程序开始时的初始化操作
{
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);
} 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(); //画地图
while()
{
drawSnake(); //画蛇
Sleep(WAIT_TIME); //等待一段时间
key();
move(); //移动蛇(主要是修改蛇身数组的数据)
}
getchar();
return ;
}
c语言贪吃蛇详解3.让蛇动起来的更多相关文章
- c语言贪吃蛇详解-2.画出蛇
c语言贪吃蛇详解-2.画出蛇 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 蛇的身 ...
- c语言贪吃蛇详解1.画出地图
c语言贪吃蛇详解-1.画出地图 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家c语言未学完,这个教程只涉及数组和函数等知识点. 首先 ...
- c语言贪吃蛇详解5.GameOver功能与显示成绩
c语言贪吃蛇详解5.GameOver功能与显示成绩 以前我们已经做出来了一个能吃东西变长的蛇.不过它好像不会死... 现在就来实现一下game over的功能吧. 写个函数判断蛇是否撞到自己或者撞到墙 ...
- c语言贪吃蛇详解4.食物的投放与蛇的变长
c语言贪吃蛇详解4.食物的投放与蛇的变长 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家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)的意思.该函数的原型为: # ...
随机推荐
- java设计模式-工厂模式(springweb为例子)
一般而言,工厂模式分为3种,简单工厂模式,工厂方法模式,抽象工厂模式.这三种工厂模式逐层深入吧. 一,从springWeb.jar包使用抽象工厂模式的一个例子聊起 之前对spring各种痴迷,所以在需 ...
- LeetCode 461. Hamming Distance (汉明距离)
The Hamming distance between two integers is the number of positions at which the corresponding bits ...
- 读书笔记-你不知道的JS中-promise(2)
继续填坑 模式 考虑下面的代码: function fn(x) { //do something return new Promise(function(resolve, reject) { //调用 ...
- 使用SQL Server2014作业定时执行无参的存储过程
一.存储过程 1.新建存储过程 按照下图找到存储过程,右键“新建”选择”存储过程” 此时在右侧会打开一个窗口,如下图. 2.填写创建存储过程语句 填写存储过程名称,因为是无参的存储过程,所以把参数部分 ...
- java语言插入数组中一个数,仍然能够实现排序
package com.llh.demo; import java.util.Scanner; /** * * @author llh * */ public class Demo16 { /* * ...
- A. 拼音魔法
A. 拼音魔法 Time limit per test: 1.0 seconds Time limit all tests: 1.0 seconds Memory limit: 256 megabyt ...
- Be the Winner
Be the Winner Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- HDU Today
HDU Today Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Su ...
- box-shadow + animation 实现loading
.loading{ width:3px; height:3px; border-radius:100%; margin-left:20px; box-shadow:0 -10px 0 1px #333 ...
- Python爬虫入门:爬虫基础了解
有粉丝私信我想让我出更基础一些的,我就把之前平台的copy下来了,可以粗略看一下,之后都会慢慢出. 1.什么是爬虫 爬虫,即网络爬虫,大家可以理解为在网络上爬行的一直蜘蛛,互联网就比作一张大网,而爬虫 ...