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.让蛇动起来的更多相关文章

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

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

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

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

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

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

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

    c语言贪吃蛇详解4.食物的投放与蛇的变长 前几天的实验室培训课后作业我布置了贪吃蛇,今天有时间就来写一下题解.我将分几步来教大家写一个贪吃蛇小游戏.由于大家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. kafka 的 createDirectStream

    一入大数据深似海,脑袋不够用了,先留下只言片语. kafka api中给出2类直接获取流的接口:createStream和createDirectStream. createStream比较简单,只需 ...

  2. JSP前后台数据交互

    1.一般方法:在页面加载时载入数据 首先,新建一个包,在包内创建一个类,类实现接口Servlet-javax.servlet 然后,在web.xml文件里配置如下 <?xml version=& ...

  3. linux命令行下svn常用命令

    linux命令行下svn常用命令 1. 将文件checkout到本地目录 1 #path是服务器上的目录 2 svn checkout path 3 4 #示例 5 svn checkout svn: ...

  4. Linux系统查找

    1. which:在当前用户环境变量path指定的路径下查找可执行程序/文件. 特点:(1)只在当前用户环境变量指定的路径下查找: (2)只找出可执行程序/文件的位置: (3)查找速度非常快. 注:使 ...

  5. nginx + tomcat + redis 部署项目,解决session共享问题。

    最近自己搭了一套nginx的环境,集群部署了公司的一个项目,中间解决了session共享的问题.记录如下,以备日后查看. 1.环境 windows10 家庭中文版,jdk 7, tomcat 7.0. ...

  6. MQTT——连接报文

    学习MQTT协议.如果只是看了相关文档就认为可以了.那是一个错误的观念.笔者为了能更好的去理解MQTT协议.看了不少相关的开源Broker的项目.可惜这些项目一般都是不完全的.不过从这些项目中笔者至少 ...

  7. IIS下自定义错误页面配置的两种方式(亲测可行)--IIS服务器

    网站自定义错误页面的设置,大家应该都知道它的重要性……不多说,下面带大家一步步在IIS下设置网站自定义错误页面…… 1.首先进入你的网站主页,找到[错误页](注意是IIS下的错误页不是.NET错误页) ...

  8. R学习笔记 第五篇:数据变换和清理

    在使用R的分组操作之前,首先要了解R语言包,包实质上是实现特定功能的,预先写好的代码库(library),R拥有大量的软件包,许多包都是由某一领域的专家编写的,但并不是所有的包都有很高的质量的,在使用 ...

  9. struts2常见错误

    1.Caught exception while loading file struts-default.xml - [unknown location]解决办法:由于lib包冲突造成的,将两个解析x ...

  10. javaCountDownLatch闭锁

    package com.java.concurrent; import java.util.concurrent.CountDownLatch; /** * CountDownLatch: 闭锁,在完 ...