这一次我们应用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. vue中template的三种写法

    第一种(使用模板字符串)早期字符串拼接年代 <div id="app"></div> new Vue({ el: "#app", tem ...

  2. vue中生命周期

    1,说器生命周期,总觉得有熟悉,又陌生,直到看到一道面试题,问父子组件的生命周期的执行顺序,我擦,真没太注意啊,不知道. 2,网上搜了一下,说法是有点像洋葱圈的形式,由外到内,在到外,因为就像一个盒子 ...

  3. [工具推荐]004.EXE签名工具SignTool使用教程

    数字证书,真是个神奇的东西,可以保证软件不被修改,可以表明文件的发布日期,最重要的,可以很大程度的减少杀毒软件的误报,当然,这就要使用可信任的机构颁发的证书了. 现在要说的不是申请证书,而是如何制作自 ...

  4. 安全性只是辅助效果?解读DevSecOps的核心动机

    DevSecOps背后的思想仅是对DevOps的扩展.就像开发人员以瀑布式开发风格将项目扔给运营团队以使其在生产中工作一样,即使使用“ DevOps”,安全性也与应用程序开发或运营完全分开. DevS ...

  5. 跨域解决方案 - JSONP

    目录 1. 定义 2. JSONP 解决跨域 3. 应用场景 4. 代码演示 1. 定义 在HTML 中, script 标签有两个个性质: script 标签可以不受同源策略的限制去访问服务器资源, ...

  6. Vuex原理实现

    Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式.它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化. 思考问题 Vuex 只在更实例引入了,那么 ...

  7. 【Spring】JdbcTemplate的使用,查询,增、删、改

    数据库名:taobaodb 配置文件: JdbcTemplate主要提供以下五类方法: execute方法:可以用于执行任何SQL语句,一般用于执行DDL语句: update方法及batchUpdat ...

  8. 从0开始探究vue-双向绑定原理

    理解 vue是一个非常优秀的框架,其优秀的双向绑定原理,mvvm模型,组件,路由解析器等,非常的灵活方便,也使开发者能够着重于数据处理,让开发者更清晰的设计自己的业务. 双向绑定,就是数据变化的时候, ...

  9. Java实现 LeetCode 264 丑数 II(二)

    264. 丑数 II 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, ...

  10. 第七届蓝桥杯JavaC组省赛真题

    解题代码部分来自网友,如果有不对的地方,欢迎各位大佬评论 题目1.有奖竞猜 题目描述 小明很喜欢猜谜语. 最近,他被邀请参加了X星球的猜谜活动. 每位选手开始的时候都被发给777个电子币. 规则是:猜 ...