这一次我们应用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语言版本)的更多相关文章

  1. 小项目特供 贪吃蛇游戏(基于C语言)

    C语言写贪吃蛇本来是打算去年暑假写的,结果因为ACM集训给耽搁了,因此借寒假的两天功夫写了这个贪吃蛇小项目,顺带把C语言重温了一次. 是发表博客的前一天开始写的,一共写了三个版本,第一天写了第一版,第 ...

  2. 【C语言项目】贪吃蛇游戏(上)

    目录 00. 目录 01. 开发背景 02. 功能介绍 03. 欢迎界面设计 3.1 常用终端控制函数 3.2 设置文本颜色函数 3.3 设置光标位置函数 3.4 绘制字符画(蛇) 3.5 欢迎界面函 ...

  3. 贪吃蛇游戏——C语言双向链表实现

    采用了双向链表结点来模拟蛇身结点: 通过C语言光标控制函数来打印地图.蛇身和食物: /************************** *************************** 贪吃 ...

  4. Love2D游戏引擎制作贪吃蛇游戏

    代码地址如下:http://www.demodashi.com/demo/15051.html Love2D游戏引擎制作贪吃蛇游戏 内附有linux下的makefile,windows下的生成方法请查 ...

  5. 用C++实现的贪吃蛇游戏

    我是一个C++初学者,控制台实现了一个贪吃蛇游戏. 代码如下: //"贪吃蛇游戏"V1.0 //李国良于2016年12月29日编写完成 #include <iostream& ...

  6. Qt 学习之路 2(34):贪吃蛇游戏(4)

    Qt 学习之路 2(34):贪吃蛇游戏(4) 豆子 2012年12月30日 Qt 学习之路 2 73条评论 这将是我们这个稍大一些的示例程序的最后一部分.在本章中,我们将完成GameControlle ...

  7. 基于React的贪吃蛇游戏的设计与实现

    代码地址如下:http://www.demodashi.com/demo/11818.html 贪吃蛇小游戏(第二版) 一年半前层用react写过贪吃蛇小游戏https://github.com/ca ...

  8. Linux平台下贪吃蛇游戏的运行

    1.参考资料说明: 这是一个在Linux系统下实现的简单的贪吃蛇游戏,同学找帮忙,我就直接在Red Hat中调试了一下,参考的是百度文库中"maosuhan"仁兄的文章,结合自己的 ...

  9. 使用Love2D引擎开发贪吃蛇游戏

    今天来介绍博主近期捣腾的一个小游戏[贪吃蛇],贪吃蛇这个游戏相信大家都不会感到陌生吧.今天博主将通过Love2D这款游戏引擎来为大家实现一个简单的贪吃蛇游戏,在本篇文章其中我们将会涉及到贪吃蛇的基本算 ...

随机推荐

  1. SVM——支持向量机(完整)

    最基本的SVM(Support Vector Machine)旨在使用一个超平面,分离线性可分的二类样本,其中正反两类分别在超平面的一侧.SVM算法则是要找出一个最优的超平面. 线性可分SVM 优化函 ...

  2. MyBatis中的命名空间namespace的作用

    1.定义mapper接口,面向接口编程. 2.在大型项目中,可能存在大量的SQL语句,这时候为每个SQL语句起一个唯一的标识(ID)就变得并不容易了.为了解决这个问题,在MyBatis中,可以为每个映 ...

  3. web自动化之执行js脚本

    from selenium import webdriver from selenium.webdriver.support.wait import WebDriverWait from seleni ...

  4. GC总结

    概述 GC(Garbage Collection),需要完成的3件事 哪些内存需要回收? 什么时候回收? 如何回收? 为什么需要了解GC和内存分配?更好的监控和调节 排查各种内存溢出,内存泄漏 避免G ...

  5. 2018京东校招Java笔试题

    相比阿里巴巴,京东的题都是考研基础题,加上一点java基础知识和linux命令. 1. 单选题(19道题,每题2分): 1)4个并发进程都需要5个同类资源,则至少需要多少个资源,才不会导致死锁? 2) ...

  6. ASP.NET Core WebAPI实现本地化(单资源文件)

    在Startup ConfigureServices 注册本地化所需要的服务AddLocalization和 Configure<RequestLocalizationOptions> p ...

  7. Java 第十一届 蓝桥杯 省模拟赛 凯撒密码加密

    凯撒密码加密 题目 问题描述 给定一个单词,请使用凯撒密码将这个单词加密. 凯撒密码是一种替换加密的技术,单词中的所有字母都在字母表上向后偏移3位后被替换成密文.即a变为d,b变为e,-,w变为z,x ...

  8. Java实现 蓝桥杯算法提高 求最大值

    算法提高 求最大值 时间限制:1.0s 内存限制:256.0MB 问题描述 给n个有序整数对ai bi,你需要选择一些整数对 使得所有你选定的数的ai+bi的和最大.并且要求你选定的数对的ai之和非负 ...

  9. Java实现 LeetCode 524 通过删除字母匹配到字典里最长单词(又是一道语文题)

    524. 通过删除字母匹配到字典里最长单词 给定一个字符串和一个字符串字典,找到字典里面最长的字符串,该字符串可以通过删除给定字符串的某些字符来得到.如果答案不止一个,返回长度最长且字典顺序最小的字符 ...

  10. Java实现 LeetCode 33 搜索旋转排序数组

    33. 搜索旋转排序数组 假设按照升序排序的数组在预先未知的某个点上进行了旋转. ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] ). 搜索一个给定的目标值, ...