基于AT89C51单片机的贪吃蛇游戏代码设计

该项目设计硬件平台选择了简单易用的at89c51单片机,显示屏选择的是ampire128x64液晶屏幕显示器,按照一定顺序连接后,如下图:
软件方面采用了c51编写代码,代码编写模块如下图:
除去网上已有的图形驱动代码外,其核心代码主要为游戏处理、信息处理和按键处理。
编写游戏的功能代码,先定义游戏的数据结构和常量。
#define uchar unsigned char
#define uint unsigned int
sbit p00 = p0 ^ 0; // 右
sbit p01 = p0 ^ 1; // 左
sbit p02 = p0 ^ 2; // 上
sbit p03 = p0 ^ 3; // 下
sbit p04 = p0 ^ 4; // 开始/暂停
#define right 1
#define left 2
#define up 3
#define down 4
#define erase 0
#define rewritten 1
#define no 0
#define yes 1
#define false 0
#define true 1
bit esc = true; // 开始 暂停标志位
uchar gamespeed = 20; // 游戏速度调节
uchar level = 1; // 难度
uchar s1[] = “1”; // 保存显示难度的汉子
// uchar = unsigned char
// 食物的结构体
struct food
{
uchar x; // 食物的横坐标
uchar y; // 食物的纵坐标
}food;
// 贪吃蛇主体的结构体
struct snake
{
uchar x[39];
uchar y[39];
uchar node; // 蛇的节数
uchar direction; // 蛇移动方向
}snake;
其中食物的数据结构采用结构体定义,两个unsigned char变量分别定义为食物的横纵坐标;蛇的身体定义为长度最大值为39的数组,游戏中贪吃蛇长度达到39,游戏通关结束。
游戏处理模块为贪吃蛇在游玩过程中遇到的需要被处理的情况,主要实现的功能包括以下四个方面,即移动、食物、死亡和更新。
移动
流程与运行仿真如上图,按下“开始”键后,游戏开始启动,初始化完毕后,屏幕会显示“贪吃蛇”、“食物”和游戏的边界框,“贪吃蛇”在固定的周期内会向前移动一格,此时“贪吃蛇”身体从尾巴至头部每一个后序节点会向前序节点移动,后序节点移动完毕后,头部会根据此时按键的方向对相应的横纵坐标进行加减。
食物
“贪吃蛇”向前移动后,此时要判断蛇头是否与“食物”的横纵坐标一一对应,如果不是则退出该模块,进入下一模块;如果是,则“贪吃蛇”的节数增加一。节数增加后判断蛇的节数是否为10的倍数,如果是则提升“贪吃蛇”的等级,并对游戏增加难度,加快贪吃蛇移动的速度。退出了游戏升级的模块后将进入创建新的“食物”模块,为了避免“食物”与“贪吃蛇”的节点坐标重复,“食物”在创建后要与“贪吃蛇”的每个节点的坐标一一比较,如果重合则重新创建“食物”的坐标,直到创建成功为止。
结束
“贪吃蛇”在向前移动后,有可能撞到墙壁或自己的身体,也有可能吃到食物,或者只是向前移动一格,移动完毕后如果吃到了食物,如果“贪吃蛇”的node(节数)达到了最大值(39),那么玩家将会通关游戏,并且游戏退出,此时需要判断游戏是否结束,一共有三种判别,前两种为失败结局,即撞到了墙壁或自己的身体,最后一种为通关结局,将在屏幕上打印“congratulations”祝贺玩家成功通关游戏。
更新
游戏在判断结束后,如果没有结束,那么更新屏幕上的“贪吃蛇”的状态。
四个模块的核心代码如下:
// 运行游戏的具体实现
void gameplay(void)
{
bit hero = false; // 完成游戏
while(1) {
ea = 1;
if (esc == false) {
gamespeed = 20;
level = 1;
s1[0] = ‘1’;
start();
}
while(esc == false) {
ea = 1;
move(); // 贪吃蛇向前移动
if(snake.x[0] == food.x && snake.y[0] == food.y) { // 判断蛇头吃到食物以后
rectarea(food.x, food.y, food.x + 2, food.y + 2, erase); // 把画面上的食物去掉
snake.node++; // 蛇的身体长一节
if (increaselevel()) { // 根据难度变化判断是否要重新开始
restart();
continue;
}
while (!createnewfood()) { // 如果创建失败则继续创建,直到成功
;
}
rectarea(food.x, food.y, food.x + 2, food.y + 2, rewritten); // 在画面上显示食物
}
if (isover()) { // 判断游戏是否结束
if(snake.node == 39)
hero = true;
}
esc = true;
hero = gameover(hero);
break;
}
// 打印贪吃蛇
updatasnakehead(); // 更新头部
delay(gamespeed); // 速度设置
erasesnaketail(); // 删除尾巴
} // 退出循环(!esc)
}
}
其余的函数代码编写如下:
// 随机数产生
int rand(void)
{
int i;
i = ((th0 《《 8) “ tl0) & 0x7fff;
return(i);
}
// 延时
void delay(uchar ms) // 10毫秒
{
uchar i, j, k;
while(ms--) {
for(i = 5; i 》 0; i--) {
for(j = 4; j 》 0; j--) {
for(k = 248; k 》 0; k--)
;
}


GGII:2022中国锂电池隔膜产能及区域分布详解
打造最小可行AI产品的详细方案
助听器电池选用
Google手机发布 售价仅1200元
模拟量采集模块的产品功能有哪些
基于AT89C51单片机的贪吃蛇游戏代码设计
全自动生化分析仪需要定标?为什么?
中国家电高端转型迎来关键期 海信电视居中国家电第一
相约首都丨4月13日-15日,南京纳特通信邀您参加第十届世界雷达博览会
新烙铁头的保养方法及注意事项
激光位移传感器怎么用?激光位移传感器安装方式
数字信号与仿真信号干扰的PCB设计技巧
智慧灯杆助力拉萨“智慧城市”建设
学技术 | USB Type-C 端口的电气风险及防范方式
越来越多的投资者正在选择比特币来对冲经济的不稳定性和通货膨胀
如何设计出高精度和隔离模拟输出模块的系统
集特与摩尔线程完成多项产品兼容互认证
解读SPCE061A智能小车语音识别系统电路
智慧公安应急指挥系统搭建指挥调度系统开发
区块链金融未来可期