基于51单片机的贪吃蛇小程序(8*8LED点阵实现)

2020-03-24来源: eefocus关键字:51单片机  贪吃蛇小程序  8*8  LED点阵

一直很想写一个贪吃蛇的小程序,这两天终于抽空完成了,这里把我的思路分享给大家,仅供参考!


代码如下:

先放段主函数压压惊


void main()

{

x_s[0] = 2;    //初始化三个点

y_s[0] = 3;

x_s[1] = 3;

y_s[1] = 3;

x_s[2] = 4;

y_s[2] = 3;

Timer0Init();

suijishu();

while(1)

{

   appear();

   if(flag_a) //蛇身移动位

   {

   if(!flag_s)   //蛇身死亡位

    {

auto_move();

HC_595(0x00);     //消隐

P0 = 0xFF; //消隐

shensi();

chiguo();

delay(1000);

flag_a = 0;

}

}

}

}


/*

本程序核心程序:

appear函数,显示蛇身

auto_move函数,控制蛇身惯性移动 ,同时里面附了判断是否穿墙的一段代码

direction函数,控制蛇的移动方向

suijishu(随机数)函数,随机生成果子

chiguo(吃果)函数,判断是否吃到果子

shensi(身死)函数,碰到自己身体则死亡


*/


#include

#include

#include

#define M 12 //难度系数,12中等,越小越难

typedef unsigned char uchar;

typedef unsigned int uint;


sbit SRCLK = P3^6;   //595芯片

sbit RCLK = P3^5;

sbit SER = P3^4;


sbit LSA=P2^2; //138译码器

sbit LSB=P2^3;

sbit LSC=P2^4;


uchar code smgduan[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,

0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//显示0~F的值


//定义方向键

sbit K1 = P3^1; //上 

sbit K2 = P3^0; //下 

sbit K3 = P3^2; //左

sbit K4 = P3^3; //右 


uchar m; //储存果子的X坐标

uchar n; //储存果子的Y坐标


uchar snakelong = 3;    //初始化蛇长

char flag_x = 1;    //初始化运动方向

char flag_y = 0;    //初始化运动方向

uchar flag_gg = 0;    //生成果子的重复性标志

uchar flag_c = 0, sheshen = 0;   //吃果标志

uchar flag_a = 0; //移动标志

uchar flag_s = 0; //死亡标志

uchar x_s[32] = {0},y_s[32] = {0};    //定义蛇的最大长度

uchar code X_[8] = {0x7F,0xBF,0xDF,0xEF,0xF7,0xFB,0xFD,0xFE};  

uchar code Y_[8] = {0x80,0x40,0x20,0x10,0x08,0x04,0x02,0x01};    


/*  

y1 //定义x,y坐标

y2   //如,我想让(4,2)亮,只需要P0 = X_[4];HC_595 = Y_[2];

y3

y3

y5

y6

y7

y8

  x1    x2   x3   x4  x5   x6   7   x8

*/


一个小的延时函数:



void suijishu();    //提前定义随机数生成函数

void delay(uint i)    //简单的delay函数,对于51单片机,delay(1)约等于9微秒

{

while(i--);

}


定时器初始化:



void Timer0Init()   //定时器初始化函数    模式1,16位定时器

{

TMOD|=0X01;//选择为定时器0模式,工作方式1,仅用TR0打开启动。


TH0=0XFC; //给定时器赋初值,定时1ms

TL0=0X18;

ET0=1;//打开定时器0中断允许

EA=1;//打开总中断

TR0=1;//打开定时器

}


两个显示函数,分别控制点阵和数码管;


void HC_595(uchar date) //595例程,这里就不详细解释了

{

uchar a;

SRCLK = 0;

RCLK = 0;


for(a=0;a<8;a++)

{

SER = date>>7;

date<<=1;

SRCLK = 1;

_nop_();

_nop_();

SRCLK = 0;

}

RCLK = 1;

_nop_();

_nop_();

RCLK = 0;

}


void display() //数码管显示,用来显示蛇长

{

uchar i,b=2;

HC_595(0x00);     //消隐

_nop_();

P0=0x00;   //因为我用的板子上数码管和点阵共用的P0口,所以这里消隐的时候需要特别注意。

_nop_();

for(i=0;i<2;i++)

{

while(b) {    //这里循环两次是为了让数码管显示更清楚一点。

switch(i) //位选,选择点亮的数码管

{

case(0):

LSA=0;LSB=0;LSC=0; P0=smgduan[(snakelong%10)];delay(10);P0=0x00;break;

case(1):

LSA=1;LSB=0;LSC=0; P0=smgduan[(snakelong/10)];delay(10);P0=0x00;break;

}

_nop_();

b--;

}

}

LSA=1;LSB=1;LSC=1; P0=0xFF; //还是为了消隐。以一个数码管常亮为代价,以求不影响点阵和数码管显示蛇长      

}


显示蛇身,核心函数之一:


void appear() // viod 蛇身出现     

{

/* x_s[0] = 2; //初始化三个点

y_s[0] = 3;

x_s[1] = 3;

y_s[1] = 3;

x_s[2] = 4;

y_s[2] = 3; */

     

//uint t = 150; //t越小越难,越大越简单 //这里也可以控制难度,经测试也有效。

//while(t)

//{

uchar i;

for(i=0;i

{

if((x_s[i]!=0)&&(y_s[i]!=0))

{

P0 = X_[x_s[i]-1];

HC_595(Y_[y_s[i]-1]);

delay(100);    //延时

P0 = 0xFF; //消隐

HC_595(0x00);     //消隐

}

}

// t--;

//}


}


自动惯性移位函数:


void auto_move() // viod 自动移动   核心程序之一

{

uchar i;

for(i=0;i<(snakelong-1);i++)

{

x_s[i] = x_s[i+1];

y_s[i] = y_s[i+1];

}

x_s[snakelong-1] += flag_x;

y_s[snakelong-1] += flag_y;


if(x_s[snakelong-1]>8) // void 穿墙判断  

x_s[snakelong-1] = 1;

else if(x_s[snakelong-1]<1)

x_s[snakelong-1] = 8;

if(y_s[snakelong-1]>8)

y_s[snakelong-1] = 1;

else if(y_s[snakelong-1]<1)

y_s[snakelong-1] = 8;

}


键盘扫描(独立键盘),控制蛇的移动方向:


void direction() // void 方向控制 

{

if(!K1) 

{

delay(1000);

if(!K1)

{

if(flag_y == 1)    //是否正在下降 ,若是,则点击上升无用。

{flag_y = 1;}

   else

    {

flag_x = 0;

flag_y = -1; //shang

}

}

}

else if(!K2) 

{

delay(1000);

if(!K2)

{

if(flag_y == -1)    //是否正在上升

{flag_y = -1;}

else

{

flag_x = 0;

flag_y = 1; //xia

}

}

}

else if(!K3) 

{

delay(1000);

if(!K3)

{

if(flag_x == 1)    //是否正在右移

{flag_x = 1;}

else

{

flag_x = -1; //zuo

flag_y = 0;

}

}

}

else if(!K4) 

{

delay(1000);

if(!K4)

{

if(flag_x == -1)    //是否正在左移

{flag_x = -1;}

else

{

flag_x = 1; //you

flag_y = 0;

}

}

}

}


吃果函数,吃到果子蛇头增长


void chiguo() // void 吃果判断

{

if(((x_s[snakelong-1]+flag_x)==m)&&((y_s[snakelong-1]+flag_y)==n)  )

{ flag_c = 1; }

else if(((x_s[snakelong-1]+flag_x)==m)&&((y_s[snakelong-1]+flag_y)==n)  ) //重复两次防止程序bug——穿过果子但不去吃     

{ flag_c = 1; }

else if(((x_s[snakelong-2]+flag_x)==m)&&((y_s[snakelong-2]+flag_y)==n)  ) //放宽吃果判定,以防止出现bug——穿过果子但不去吃               

{ flag_c = 1; sheshen = 1;    }

else if(((x_s[snakelong-3]+flag_x)==m)&&((y_s[snakelong-3]+flag_y)==n)  )

{ flag_c = 1; sheshen = 2;    }

   

if(flag_c)

{

flag_c = 0;

TR0 = 0;//关闭定时器

snakelong++;

if(sheshen == 1)

{

x_s[snakelong-1] = m+flag_x;

y_s[snakelong-1] = n+flag_y;

sheshen = 0;

}

else if(sheshen == 2)

{

x_s[snakelong-1] = m+2*flag_x;

y_s[snakelong-1] = n+2*flag_y;

sheshen = 0;

}

else

{ //这才是吃果的正常情况

    x_s[snakelong-1] = m;

y_s[snakelong-1] = n;

sheshen = 0;

}

        suijishu();   //再生成一个果子

P0 = 0xFF; //消隐

HC_595(0x00);     //消隐

appear();

appear();

P0 = 0xFF; //消隐

HC_595(0x00);     //消隐

TR0=1;//打开定时器

}

}


碰到自己身体死亡函数


void shensi() // void 身死    碰到自己身死

{

uchar b,x_1,y_1;

if(snakelong>4)

{

for(b=0;b

  {

if(((x_s[snakelong-1])==x_s[b])&&((y_s[snakelong-1])==y_s[b])  )  

{

flag_s = 1; 

}

  }

if(flag_s == 1)

{

P0 = 0x00;HC_595(0xFF);delay(2000);    //闪烁两次

P0 = 0xFF;HC_595(0x00);delay(2000);

P0 = 0x00;HC_595(0xFF);delay(3000);

P0 = 0xFF;HC_595(0x00);delay(1000);

x_1 = x_s[snakelong-1] ;

y_1 = y_s[snakelong-1] ;      //卡在死亡状态的下一刻  

  snakelong++;

x_s[snakelong-1] = x_1 + flag_x;

y_s[snakelong-1] = y_1 + flag_y;

}


}

}


随机生成果子:



void suijishu() //随机数,生成果子

{

uchar b,m_1,n_1;

do{

flag_gg = 0;

m_1 = (rand()%8)+1;    

n_1 = (rand()%8)+1;

for(b=0;b

  {

if((m_1==x_s[b])&&(n_1==y_s[b])) //如果果子和蛇身有重复,则flag_gg置1,再生成一次                    

flag_gg = 1;

  }

}while(flag_gg);

m = m_1;

n = n_1;

}


主函数:


void main()

{

x_s[0] = 2;    //初始化三个点

y_s[0] = 3;

x_s[1] = 3;

y_s[1] = 3;

x_s[2] = 4;

y_s[2] = 3;

Timer0Init();

suijishu();

while(1)

{

   appear();

   if(flag_a) //蛇身移动位

   {

   if(!flag_s)   //蛇身死亡位

    {

auto_move();

HC_595(0x00);     //消隐

P0 = 0xFF; //消隐

shensi();

chiguo();

delay(1000);

flag_a = 0;

}

}

}

}


/*

果子的函数放在了中断里,1ms一次进行显示,同时键盘也是1mS一次进行扫描

通过调节M的大小来实现蛇的延时移动。

*/


void guozi() interrupt 1 // void 果子出现

{

uint a;

uchar i=3;

a++;

direction();   //用中断去扫描键盘,1mS/次

P0 = 0xFF; //消隐

HC_595(0x00);     //消隐

while(i)   //重复以增加亮度

{

P0 = X_[m-1];

HC_595(Y_[n-1]);

i--;

}

delay(100);

P0 = 0xFF; //消隐 ,因为我用的板子上数码管和点阵共用的P0口,所以这里消隐的时候需要特别注意

HC_595(0x00);     //消隐


_nop_();

P0=0x00; //数码管

[1] [2]
关键字:51单片机  贪吃蛇小程序  8*8  LED点阵 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic492469.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:基于51单片机的简单计算器
下一篇:51单片机的8x8点阵贪吃蛇

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

9课:单片机数据传递类指令
(R1)=12H(A)=30H(34H)=34HMOV 45H,34HMOV DPTR,#6712HMOV 12H,DPHMOV R0,DPLMOV A,@R0(45H)=34H(DPTR)=6712H(12H)=67H(R0)=12H(A)=67H说明:用括号括起来代表内容,如(23H)则代表内部RAM23H单元中的值,(A)则代表累加器A单元中的值。进入DOS状态,进入WAVE所在的目录,例D:WAVE键入MCS51,出现如下画面<单片机数据传递指令>图1按File->Open,出现对话框后,在Name处输入一个文件名(见图2),如果是下面列表中已存在的,则打开这个文件,如果不存在这个文件,则新建一个文件(见图
发表于 2020-03-10
9课:单片机数据传递类指令
8课:单片机寻址方式与指令系统
通过前面的学习,我们已经了解了单片机内部的结构,并且也已经知道,要控制单片机,让它为我们干学,要用指令,我们已学了几条指令,但很零散,从现在开始,我们将要系统地学习8051单片机的指令部份。一、概述1、指令的格式我们已知,要让计算机做事,就得给计算机以指令,并且我们已知,计算机很“笨”,只能懂得数字,如前面我们写进机器的75H,90H,00H等等,所以指令的第一种格式就是机器码格式,也说是数字的形式。但这种形式实在是为难我们人了,太难记了,于是有另一种格式,助记符格式,如MOV P1,#0FFH,这样就好记了。 这两种格式之间的关系呢,我们不难理解,本质上它们完全等价,只是形式不一样而已。2、汇编我们写指令使用汇编格式,而计算机
发表于 2020-03-10
7课:单片机的特殊功能寄存器
通过前面的学习,我们已知单片机的内部有ROM、有RAM、有并行I/O口,那么,除了这些东西之外,单片机内部究竟还有些什么,这些个零碎的东西怎么连在一起的,让我们来对单片机内部的寄存器作一个完整的功能分析吧!下图中我们能看出,在51单片机内部有一个CPU用来运算、控制,有四个并行I/O口,分别是P0、P1、P2、P3,有ROM,用来存放程序,有RAM,用来存放中间结果,此外还有定时/计数器,串行I/O口,中断系统,以及一个内部的时钟电路。在一个51单片机的内部包含了这么多的东西。对上面的图进行进一步的分析,我们已知,对并行I/O口的读写只要将数据送入到对应I/O口的锁存器就能了,那么对于定时/计数器,串行I/O口
发表于 2020-03-10
7课:单片机的特殊功能寄存器
6课:单片机并行口结构
”。2、输入结构这是并行口的一位的输出结构示意图,再看,除了输出之外,还有两根线,一根从外部管脚接入,另一根从锁存器的输出接出,分别标明读管脚和读锁存器。这两根线是用于从外部接收信号的,为什么要两根呢?原来,在51单片机中输入有两种方式,分别称为‘读管脚’和‘读锁存器’,第一种方式是将管脚作为输入,那是真正地从外部管脚读进输入的值,第二种方式是该管脚处于输出状态时,有时需要改变这一位的状态,则并不需要真正地读管脚状态,而只是读入锁存器的状态,然后作某种变换后再输出。请注意输入结构图,如果将这一根引线作为输入口使用,我们并不能保证在任何时刻都能得到正确的结果(为什么?)参考图2输入示意图。接在外部的开关如果打开,则应当是输入1,而如果闭合开关
发表于 2020-03-10
6课:单片机并行口结构
5课:单片机延时程序分析
是多长时间吧。设一个单片机工作于12M晶体震荡器,它的时钟周期是1/12(微秒)。它的一个机器周期是12*(1/12)也就是1微秒。(请计算一个工作于6M晶体震荡器的单片机,它的机器周期是多少)。MCS-51单片机的所有指令中,有一些完成得比较快,只要一个机器周期就行了,有一些完成得比较慢,得要2个机器周期,还有两条指令要4个机器周期才行。这也不难再解,不是吗?我让你扫地的执行要完成总得比要你完成擦黑板的指令时间要长。为了恒量指令执行时间的长短,又引入一个新的概念:指令周期。所谓指令周期就是指执行一条指令的时间。INTEL对每一条指令都给出了它的指令周期数,这些数据,大部份不需要我们去记忆,但是有一些指令是需要记住的,如DJNZ指令是双周
发表于 2020-03-10
5课:单片机延时程序分析
单片机模拟I2C总线及24C02(I2C EEPROM)读写实例
;*//************************************************************************//* Name:AT24C02存储器的读写程序,用到I2C总线,含相对独立的I2C总线读写函数 *//* Language: C51单片机编程语言                                          *//* Platform: Win98,Intel Celeron 433 Processor,伟福仿真器,仿真8751     *//* Author
发表于 2020-03-10
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2020 EEWORLD.com.cn, Inc. All rights reserved