贪吃蛇游戏(printf输出C语言版本)
这一次我们应用printf输出实现一个经典的小游戏—贪吃蛇,主要难点是小蛇数据如何存储、如何实现转弯的效果、吃到食物后如何增加长度。
1 构造小蛇
首先,在画面中显示一条静止的小蛇。二维数组canvas[High][Width]的对应元素,值为0输出空格,-1输出边框#,1输出蛇头@,大于1的正数输出蛇身*。startup()函数中初始化蛇头在画布中间位置(canvas[High/2][Width/2] = 1;),蛇头向左依次生成4个蛇身(for (i=1;i<=4;i++) canvas[High/2][Width/2-i] = i+1;),元素值分别为2、3、4、5。

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include <windows.h>
//C语言自学网
#define High 20 // 游戏画面尺寸
#define Width 30 // 全局变量
int canvas[High][Width] = {}; // 二维数组存储游戏画布中对应的元素
// 0为空格,-1为边框#,1为蛇头@,大于1的正数为蛇身* void gotoxy(int x,int y) //光标移动到(x,y)位置
{
HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);
COORD pos;
pos.X = x;
pos.Y = y;
SetConsoleCursorPosition(handle,pos);
} void startup() // 数据初始化
{
int i,j; // 初始化边框
for (i=;i<High;i++)
{
canvas[i][] = -;
canvas[i][Width-] = -;
}
for (j=;j<Width;j++)
{
canvas[][j] = -;
canvas[High-][j] = -;
} // 初始化蛇头位置
canvas[High/][Width/] = ;
// 初始化蛇身,画布中元素值分别为2,3,4,5....
for (i=;i<=;i++)
canvas[High/][Width/-i] = i+;
} void show() // 显示画面
{
gotoxy(,); // 光标移动到原点位置,以下重画清屏
int i,j;
for (i=;i<High;i++)
{
for (j=;j<Width;j++)
{
if (canvas[i][j]==)
printf(" "); // 输出空格
else if (canvas[i][j]==-)
printf("#"); // 输出边框#
else if (canvas[i][j]==)
printf("@"); // 输出蛇头@
else if (canvas[i][j]>)
printf("*"); // 输出蛇身*
}
printf("\n");
}
} void updateWithoutInput() // 与用户输入无关的更新
{
} void updateWithInput() // 与用户输入有关的更新
{
} int main()
{
startup(); // 数据初始化
while () // 游戏循环执行
{
show(); // 显示画面
updateWithoutInput(); // 与用户输入无关的更新
updateWithInput(); // 与用户输入有关的更新
}
return ;
}
2 小蛇自动移动
实现小蛇的移动是贪吃蛇游戏的难点,下图列出了小蛇分别向右、向上运动后,对应二维数组元素值的变化,从中我们可以得出实现思路。

假设小蛇元素为54321,其中1为蛇头、5432为蛇身、最大值5为蛇尾。首先将所有大于0的元素加1,得到65432;将最大值6变为0,即去除了原来的蛇尾;再根据对应的移动方向,将2对应方向的元素由0变成1;如此即实现了小蛇的移动。小蛇向上移动的对应流程如图所示。

定义变量int moveDirection表示小蛇的移动方向,值1、2、3、4分别表示小蛇向上、下、左、右方向移动,小蛇移动实现在moveSnakeByDirection()函数中。
#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include <windows.h>
//C语言自学网
#define High 20 // 游戏画面尺寸
#define Width 30 // 全局变量
int moveDirection; // 小蛇移动方向,上下左右分别用1,2,3,4表示
int canvas[High][Width] = {}; // 二维数组存储游戏画布中对应的元素
// 0为空格0,-1为边框#,1为蛇头@,大于1的正数为蛇身* void gotoxy(int x,int y) //光标移动到(x,y)位置
{
HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);
COORD pos;
pos.X = x;
pos.Y = y;
SetConsoleCursorPosition(handle,pos);
} // 移动小蛇
// 第一步扫描数组canvas所有元素,找到正数元素都+1
// 找到最大元素(即蛇尾巴),把其变为0
// 找到=2的元素(即蛇头),再根据输出的上下左右方向,把对应的另一个像素值设为1(新蛇头)
void moveSnakeByDirection()
{
int i,j;
for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
canvas[i][j]++; int oldTail_i,oldTail_j,oldHead_i,oldHead_j;
int max = ; for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
{
if (max<canvas[i][j])
{
max = canvas[i][j];
oldTail_i = i;
oldTail_j = j;
}
if (canvas[i][j]==)
{
oldHead_i = i;
oldHead_j = j;
}
} canvas[oldTail_i][oldTail_j] = ; if (moveDirection==) // 向上移动
canvas[oldHead_i-][oldHead_j] = ;
if (moveDirection==) // 向下移动
canvas[oldHead_i+][oldHead_j] = ;
if (moveDirection==) // 向左移动
canvas[oldHead_i][oldHead_j-] = ;
if (moveDirection==) // 向右移动
canvas[oldHead_i][oldHead_j+] = ;
} void startup() // 数据初始化
{
int i,j; // 初始化边框
for (i=;i<High;i++)
{
canvas[i][] = -;
canvas[i][Width-] = -;
}
for (j=;j<Width;j++)
{
canvas[][j] = -;
canvas[High-][j] = -;
} // 初始化蛇头位置
canvas[High/][Width/] = ;
// 初始化蛇身,画布中元素值分别为2,3,4,5....
for (i=;i<=;i++)
canvas[High/][Width/-i] = i+; // 初始小蛇向右移动
moveDirection = ;
} void show() // 显示画面
{
gotoxy(,); // 光标移动到原点位置,以下重画清屏
int i,j;
for (i=;i<High;i++)
{
for (j=;j<Width;j++)
{
if (canvas[i][j]==)
printf(" "); // 输出空格
else if (canvas[i][j]==-)
printf("#"); // 输出边框#
else if (canvas[i][j]==)
printf("@"); // 输出蛇头@
else if (canvas[i][j]>)
printf("*"); // 输出蛇身*
}
printf("\n");
}
Sleep();
} void updateWithoutInput() // 与用户输入无关的更新
{
moveSnakeByDirection();
} void updateWithInput() // 与用户输入有关的更新
{
} int main()
{
startup(); // 数据初始化
while () // 游戏循环执行
{
show(); // 显示画面
updateWithoutInput(); // 与用户输入无关的更新
updateWithInput(); // 与用户输入有关的更新
}
return ;
}
3 玩家控制小蛇移动
这一步的实现比较简单,在updateWithInput()函数中按asdw键改变moveDirection的值,然后调用moveSnakeByDirection()实现小蛇向不同方向的移动,如图所示。

void updateWithInput() // 与用户输入有关的更新
//C语言自学网
{
char input;
if(kbhit()) // 判断是否有输入
{
input = getch(); // 根据用户的不同输入来移动,不必输入回车
if (input == 'a')
{
moveDirection = ; // 位置左移
moveSnakeByDirection();
}
else if (input == 'd')
{
moveDirection = ; // 位置右移
moveSnakeByDirection();
}
else if (input == 'w')
{
moveDirection = ; // 位置上移
moveSnakeByDirection();
}
else if (input == 's')
{
moveDirection = ; // 位置下移
moveSnakeByDirection();
}
}
}
4 判断游戏失败
当小蛇和边框或自身发生碰撞时,游戏失败,如图所示。

void moveSnakeByDirection()
//C语言自学网
{
int i,j;
for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
canvas[i][j]++;
int oldTail_i,oldTail_j,oldHead_i,oldHead_j;
int max = ;
for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
{
if (max<canvas[i][j])
{
max = canvas[i][j];
oldTail_i = i;
oldTail_j = j;
}
if (canvas[i][j]==)
{
oldHead_i = i;
oldHead_j = j;
}
}
canvas[oldTail_i][oldTail_j] = ;
int newHead_i,newHead_j;
if (moveDirection==) // 向上移动
{
newHead_i = oldHead_i-;
newHead_j = oldHead_j;
}
if (moveDirection==) // 向下移动
{
newHead_i = oldHead_i+;
newHead_j = oldHead_j;
}
if (moveDirection==) // 向左移动
{
newHead_i = oldHead_i;
newHead_j = oldHead_j-;
}
if (moveDirection==) // 向右移动
{
newHead_i = oldHead_i;
newHead_j = oldHead_j+;
} // 是否小蛇和自身撞,或者和边框撞,游戏失败
if (canvas[newHead_i][newHead_j]> || canvas[newHead_i][newHead_j]==-)
{
printf("游戏失败!\n");
exit();
}
else
canvas[newHead_i][newHead_j] = ;
}
5 吃食物增加长度
增加食物,二维数组canvas[High][Width]元素值为-2时,输出食物数值’F’,如图所示。当蛇头碰到食物时,长度加一。

实现思路和2中小蛇移动类似,只需保持原蛇尾,不将最大值变为0即可,下图为小蛇向上移动吃到食物的对应流程。

#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include <windows.h>
//C语言自学网
#define High 20 // 游戏画面尺寸
#define Width 30 // 全局变量
int moveDirection; // 小蛇移动位置,上下左右分别用1,2,3,4表示
int food_x,food_y; // 食物的位置
int canvas[High][Width] = {}; // 二维数组存储游戏画布中对应的元素
// 0为空格0,-1为边框#,-2为食物F,1为蛇头@,大于1的正数为蛇身* void gotoxy(int x,int y) //光标移动到(x,y)位置
{
HANDLE handle = GetStdHandle(STD_OUTPUT_HANDLE);
COORD pos;
pos.X = x;
pos.Y = y;
SetConsoleCursorPosition(handle,pos);
} // 移动小蛇
// 第一步扫描数组canvas所有元素,找到正数元素都+1
// 找到最大元素(即蛇尾巴),把其变为0
// 找到=2的元素(即蛇头),再根据输出的上下左右方向,把对应的另一个像素值设为1(新蛇头)
void moveSnakeByDirection()
{
int i,j;
for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
canvas[i][j]++; int oldTail_i,oldTail_j,oldHead_i,oldHead_j;
int max = ; for (i=;i<High-;i++)
for (j=;j<Width-;j++)
if (canvas[i][j]>)
{
if (max<canvas[i][j])
{
max = canvas[i][j];
oldTail_i = i;
oldTail_j = j;
}
if (canvas[i][j]==)
{
oldHead_i = i;
oldHead_j = j;
}
} int newHead_i,newHead_j; if (moveDirection==) // 向上移动
{
newHead_i = oldHead_i-;
newHead_j = oldHead_j;
}
if (moveDirection==) // 向下移动
{
newHead_i = oldHead_i+;
newHead_j = oldHead_j;
}
if (moveDirection==) // 向左移动
{
newHead_i = oldHead_i;
newHead_j = oldHead_j-;
}
if (moveDirection==) // 向右移动
{
newHead_i = oldHead_i;
newHead_j = oldHead_j+;
} // 新蛇头如果吃到食物
if (canvas[newHead_i][newHead_j]==-)
{
canvas[food_x][food_y] = ;
// 产生一个新的食物
food_x = rand()%(High-) + ;
food_y = rand()%(Width-) + ;
canvas[food_x][food_y] = -; // 原来的旧蛇尾留着,长度自动+1
}
else // 否则,原来的旧蛇尾减掉,保持长度不变
canvas[oldTail_i][oldTail_j] = ; // 是否小蛇和自身撞,或者和边框撞,游戏失败
if (canvas[newHead_i][newHead_j]> || canvas[newHead_i][newHead_j]==-)
{
printf("游戏失败!\n");
Sleep();
system("pause");
exit();
}
else
canvas[newHead_i][newHead_j] = ;
} void startup() // 数据初始化
{
int i,j; // 初始化边框
for (i=;i<High;i++)
{
canvas[i][] = -;
canvas[i][Width-] = -;
}
for (j=;j<Width;j++)
{
canvas[][j] = -;
canvas[High-][j] = -;
} // 初始化蛇头位置
canvas[High/][Width/] = ;
// 初始化蛇身,画布中元素值分别为2,3,4,5....
for (i=;i<=;i++)
canvas[High/][Width/-i] = i+; // 初始小蛇向右移动
moveDirection = ; food_x = rand()%(High-) + ;
food_y = rand()%(Width-) + ;
canvas[food_x][food_y] = -;
} void show() // 显示画面
{
gotoxy(,); // 光标移动到原点位置,以下重画清屏
int i,j;
for (i=;i<High;i++)
{
for (j=;j<Width;j++)
{
if (canvas[i][j]==)
printf(" "); // 输出空格
else if (canvas[i][j]==-)
printf("#"); // 输出边框#
else if (canvas[i][j]==)
printf("@"); // 输出蛇头@
else if (canvas[i][j]>)
printf("*"); // 输出蛇身*
else if (canvas[i][j]==-)
printf("F"); // 输出食物F
}
printf("\n");
}
Sleep();
} void updateWithoutInput() // 与用户输入无关的更新
{
moveSnakeByDirection();
} void updateWithInput() // 与用户输入有关的更新
{
char input;
if(kbhit()) // 判断是否有输入
{
input = getch(); // 根据用户的不同输入来移动,不必输入回车
if (input == 'a')
{
moveDirection = ; // 位置左移
moveSnakeByDirection();
}
else if (input == 'd')
{
moveDirection = ; // 位置右移
moveSnakeByDirection();
}
else if (input == 'w')
{
moveDirection = ; // 位置上移
moveSnakeByDirection();
}
else if (input == 's')
{
moveDirection = ; // 位置下移
moveSnakeByDirection();
}
}
} int main()
{
startup(); // 数据初始化
while () // 游戏循环执行
{
show(); // 显示画面
updateWithoutInput(); // 与用户输入无关的更新
updateWithInput(); // 与用户输入有关的更新
}
return ;
}
6 思考题
1. 增加道具,吃完可以加命或减速;
2. 尝试实现双人版贪吃蛇
感谢你的阅读,请用心感悟!希望可以帮到爱学习的你!!分享也是一种快乐!!!请接力。。。
贪吃蛇游戏(printf输出C语言版本)的更多相关文章
- 小项目特供 贪吃蛇游戏(基于C语言)
C语言写贪吃蛇本来是打算去年暑假写的,结果因为ACM集训给耽搁了,因此借寒假的两天功夫写了这个贪吃蛇小项目,顺带把C语言重温了一次. 是发表博客的前一天开始写的,一共写了三个版本,第一天写了第一版,第 ...
- 【C语言项目】贪吃蛇游戏(上)
目录 00. 目录 01. 开发背景 02. 功能介绍 03. 欢迎界面设计 3.1 常用终端控制函数 3.2 设置文本颜色函数 3.3 设置光标位置函数 3.4 绘制字符画(蛇) 3.5 欢迎界面函 ...
- 贪吃蛇游戏——C语言双向链表实现
采用了双向链表结点来模拟蛇身结点: 通过C语言光标控制函数来打印地图.蛇身和食物: /************************** *************************** 贪吃 ...
- Love2D游戏引擎制作贪吃蛇游戏
代码地址如下:http://www.demodashi.com/demo/15051.html Love2D游戏引擎制作贪吃蛇游戏 内附有linux下的makefile,windows下的生成方法请查 ...
- 用C++实现的贪吃蛇游戏
我是一个C++初学者,控制台实现了一个贪吃蛇游戏. 代码如下: //"贪吃蛇游戏"V1.0 //李国良于2016年12月29日编写完成 #include <iostream& ...
- Qt 学习之路 2(34):贪吃蛇游戏(4)
Qt 学习之路 2(34):贪吃蛇游戏(4) 豆子 2012年12月30日 Qt 学习之路 2 73条评论 这将是我们这个稍大一些的示例程序的最后一部分.在本章中,我们将完成GameControlle ...
- 基于React的贪吃蛇游戏的设计与实现
代码地址如下:http://www.demodashi.com/demo/11818.html 贪吃蛇小游戏(第二版) 一年半前层用react写过贪吃蛇小游戏https://github.com/ca ...
- Linux平台下贪吃蛇游戏的运行
1.参考资料说明: 这是一个在Linux系统下实现的简单的贪吃蛇游戏,同学找帮忙,我就直接在Red Hat中调试了一下,参考的是百度文库中"maosuhan"仁兄的文章,结合自己的 ...
- 使用Love2D引擎开发贪吃蛇游戏
今天来介绍博主近期捣腾的一个小游戏[贪吃蛇],贪吃蛇这个游戏相信大家都不会感到陌生吧.今天博主将通过Love2D这款游戏引擎来为大家实现一个简单的贪吃蛇游戏,在本篇文章其中我们将会涉及到贪吃蛇的基本算 ...
随机推荐
- 【Linux】Linux常用操作
终端命令格式 command [-options] [parameter]command : 命令名,相应功能的英文单词或单词的缩写[-options]:选项,可用来对命令进行控制,也可以省略para ...
- 二,表格<table>的使用细节
如有不足请不吝赐教!最先接触的布局表格,表格灵活性,加载速度都不如<div>灵活,但是对于萌新来说再定位某些,例如div里面套文字,而刚接触html+css又不懂得那么多使用<spa ...
- ubuntu 安装 swftoos
一:下载依赖: freetype下载地址 : http://ftp.twaren.net/Unix/NonGNU/freetype/ jpegsrc:下载地址 http://www.ijg.org/f ...
- [Firefox附加组件]0001.入门
Firefox 火狐浏览器,拥有最快.最安全的上网体验,并且火狐拥有超过一万个的 扩展(add-ons),提供各种不同的扩展功能,您可以简单的下载.安装这些扩展以增强您的火狐功能,帮助您更好.更个性化 ...
- 01 . Memcached简介及部署
Memcached简介 memcached是一个自由开源,高性能,分布式内存对象存储系统 基于内存的key-valued存储,用来存储小块的任意数据(字符串,对象) 他是一个简洁的key-value存 ...
- Java实现蓝桥杯VIP算法训练 奇变的字符串
试题 算法训练 奇变的字符串 资源限制 时间限制:1.0s 内存限制:256.0MB 问题描述 将一个字符串的奇数位(首位为第0位)取出,将其顺序弄反,再放回原字符串的原位置上. 如字符串" ...
- Java实现 蓝桥杯 算法训练 数字游戏
试题 算法训练 数字游戏 资源限制 时间限制:1.0s 内存限制:256.0MB 问题描述 给定一个1-N的排列a[i],每次将相邻两个数相加,得到新序列,再对新序列重复这样的操作,显然每次得到的序列 ...
- Java实现 蓝桥杯VIP 算法提高 邮票面值设计
算法提高 邮票面值设计 时间限制:1.0s 内存限制:256.0MB 问题描述 给定一个信封,最多只允许粘贴N张邮票,计算在给定K(N+K≤13)种邮票的情况下(假定所有的邮票数量都足够),如何设计邮 ...
- 第五届蓝桥杯JavaA组国(决)赛真题
解题代码部分来自网友,如果有不对的地方,欢迎各位大佬评论 题目1.海盗分金币 有5个海盗,相约进行一次帆船比赛. 比赛中天气发生突变,他们被冲散了. 恰巧,他们都先后经过途中的一个无名的荒岛,并且每个 ...
- java实现第五届蓝桥杯幂一矩阵
幂一矩阵 天才少年的邻居 atm 最近学习了线性代数相关的理论,他对"矩阵"这个概念特别感兴趣.矩阵中有个概念叫做幂零矩阵.对于一个方阵 M ,如果存在一个正整数 k 满足 M^k ...