历史上的今天
返回首页

历史上的今天

今天是:2025年02月07日(星期五)

正在发生

2021年02月07日 | 51单片机数码管显示跑马灯程序源代码讲解

2021-02-07 来源:eefocus

基于51单片机学习板。用S1键作为控制跑马灯的方向按键,S5键作为控制跑马灯方向的加速度按键,S9键作为控制跑马灯方向的减速度按键,S13键作为控制跑马灯方向的启动或者暂停按键。记得把输出线P0.4一直输出低电平,模拟独立按键的触发地GND。

(2)实现功能:

跑马灯运行:第1个至第8个LED灯一直不亮。在第9个至第16个LED灯,依次逐个亮灯并且每次只能亮一个灯。每按一次独立按键S13键,原来运行的跑马灯会暂停,原来暂停的跑马灯会运行。用S1来改变方向。用S5和S9来改变速度,每按一次按键的递增或者递减以10为单位。

数码管显示:本程序只有1个窗口,这个窗口分成3个局部显示。8,7,6位数码管显示运行状态,启动时显示“on”,停止时显示“oFF”。5位数码管显示数码管方向,正向显示“n”,反向显示“U”。4,3,2,1位数码管显示速度。数值越大速度越慢,最慢的速度是550,最快的速度是50。

(3)源代码讲解如下:

#include "REG52.H"

#define const_voice_short 40 //蜂鸣器短叫的持续时间

#define const_key_time1 20 //按键去抖动延时的时间

#define const_key_time2 20 //按键去抖动延时的时间

#define const_key_time3 20 //按键去抖动延时的时间

#define const_key_time4 20 //按键去抖动延时的时间


void initial_myself();

void initial_peripheral();

void delay_short(unsigned int uiDelayShort);

void delay_long(unsigned int uiDelaylong);


//驱动数码管的74HC595

void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);

void display_drive(); //显示数码管字模的驱动函数

void display_service(); //显示的窗口菜单服务程序


//驱动LED的74HC595

void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);

void led_flicker_09_16(); //第9个至第16个LED的跑马灯程序,逐个亮并且每次只能亮一个.

void led_update(); //LED更新函数


void T0_time(); //定时中断函数

void key_service(); //按键服务的应用程序

void key_scan();//按键扫描函数 放在定时中断里



sbit beep_dr=P2^7; //蜂鸣器的驱动IO口

sbit key_sr1=P0^0; //对应学习板的S1键

sbit key_sr2=P0^1; //对应学习板的S5键

sbit key_sr3=P0^2; //对应学习板的S9键

sbit key_sr4=P0^3; //对应学习板的S13键


sbit key_gnd_dr=P0^4; //模拟独立按键的地GND,因此必须一直输出低电平


sbit led_dr=P3^5;



sbit dig_hc595_sh_dr=P2^0; //数码管的74HC595程序

sbit dig_hc595_st_dr=P2^1;

sbit dig_hc595_ds_dr=P2^2;


sbit hc595_sh_dr=P2^3; //LED灯的74HC595程序

sbit hc595_st_dr=P2^4;

sbit hc595_ds_dr=P2^5;



unsigned char ucKeySec=0; //被触发的按键编号


unsigned int uiKeyTimeCnt1=0; //按键去抖动延时计数器

unsigned char ucKeyLock1=0; //按键触发后自锁的变量标志


unsigned int uiKeyTimeCnt2=0; //按键去抖动延时计数器

unsigned char ucKeyLock2=0; //按键触发后自锁的变量标志


unsigned int uiKeyTimeCnt3=0; //按键去抖动延时计数器

unsigned char ucKeyLock3=0; //按键触发后自锁的变量标志



unsigned int uiKeyTimeCnt4=0; //按键去抖动延时计数器

unsigned char ucKeyLock4=0; //按键触发后自锁的变量标志


unsigned int uiVoiceCnt=0; //蜂鸣器鸣叫的持续时间计数器


unsigned char ucLed_dr1=0; //代表16个灯的亮灭状态,0代表灭,1代表亮

unsigned char ucLed_dr2=0;

unsigned char ucLed_dr3=0;

unsigned char ucLed_dr4=0;

unsigned char ucLed_dr5=0;

unsigned char ucLed_dr6=0;

unsigned char ucLed_dr7=0;

unsigned char ucLed_dr8=0;

unsigned char ucLed_dr9=0;

unsigned char ucLed_dr10=0;

unsigned char ucLed_dr11=0;

unsigned char ucLed_dr12=0;

unsigned char ucLed_dr13=0;

unsigned char ucLed_dr14=0;

unsigned char ucLed_dr15=0;

unsigned char ucLed_dr16=0;


unsigned char ucLed_update=0; //刷新变量。每次更改LED灯的状态都要更新一次。



unsigned char ucLedStep_09_16=0; //第9个至第16个LED跑马灯的步骤变量

unsigned int uiTimeCnt_09_16=0; //第9个至第16个LED跑马灯的统计定时中断次数的延时计数器


unsigned char ucLedStatus16_09=0; //代表底层74HC595输出状态的中间变量

unsigned char ucLedStatus08_01=0; //代表底层74HC595输出状态的中间变量


unsigned char ucLedDirFlag=0; //方向变量,把按键与跑马灯关联起来的核心变量,0代表正方向,1代表反方向

unsigned int uiSetTimeLevel_09_16=300; //速度变量,此数值越大速度越慢,此数值越小速度越快。

unsigned char ucLedStartFlag=1; //启动和暂停的变量,0代表暂停,1代表启动




unsigned char ucDigShow8; //第8位数码管要显示的内容

unsigned char ucDigShow7; //第7位数码管要显示的内容

unsigned char ucDigShow6; //第6位数码管要显示的内容

unsigned char ucDigShow5; //第5位数码管要显示的内容

unsigned char ucDigShow4; //第4位数码管要显示的内容

unsigned char ucDigShow3; //第3位数码管要显示的内容

unsigned char ucDigShow2; //第2位数码管要显示的内容

unsigned char ucDigShow1; //第1位数码管要显示的内容



unsigned char ucDigDot8; //数码管8的小数点是否显示的标志

unsigned char ucDigDot7; //数码管7的小数点是否显示的标志

unsigned char ucDigDot6; //数码管6的小数点是否显示的标志

unsigned char ucDigDot5; //数码管5的小数点是否显示的标志

unsigned char ucDigDot4; //数码管4的小数点是否显示的标志

unsigned char ucDigDot3; //数码管3的小数点是否显示的标志

unsigned char ucDigDot2; //数码管2的小数点是否显示的标志

unsigned char ucDigDot1; //数码管1的小数点是否显示的标志


unsigned char ucDigShowTemp=0; //临时中间变量

unsigned char ucDisplayDriveStep=1; //动态扫描数码管的步骤变量


unsigned char ucWd1Part1Update=1; //窗口1的局部1更新显示变量

unsigned char ucWd1Part2Update=1; //窗口1的局部2更新显示变量

unsigned char ucWd1Part3Update=1; //窗口1的局部3更新显示变量



//根据原理图得出的共阴数码管字模表

code unsigned char dig_table[]=

{

0x3f, //0 序号0

0x06, //1 序号1

0x5b, //2 序号2

0x4f, //3 序号3

0x66, //4 序号4

0x6d, //5 序号5

0x7d, //6 序号6

0x07, //7 序号7

0x7f, //8 序号8

0x6f, //9 序号9

0x00, //无 序号10

0x40, //- 序号11

0x73, //P 序号12

0x5c, //o 序号13

0x71, //F 序号14

0x3e, //U 序号15

0x37, //n 序号16

};


void main()

{

initial_myself();

delay_long(100);

initial_peripheral();

while(1)

{

key_service(); //按键服务的应用程序

display_service(); //显示的窗口菜单服务程序


led_flicker_09_16(); //第9个至第16个LED的跑马灯程序,逐个亮并且每次只能亮一个.

led_update(); //LED更新函数

}


}




/* 注释一:

* 由于本程序只有1个窗口,而这个窗口又分成3个局部,因此可以省略去窗口变量uWd,

* 只用三个局部变量ucWdxPartyUpdate就可以了。

*/


void display_service() //显示的窗口菜单服务程序

{



if(ucWd1Part1Update==1) //更新显示当前系统是处于运行还是暂停的状态

{

ucWd1Part1Update=0; //及时把更新变量清零,防止一直进来更新

if(ucLedStartFlag==1) //启动,显示on

{

ucDigShow8=13; //显示o

ucDigShow7=16; //显示n

ucDigShow6=10; //显示空

}

else //暂停,显示oFF

{

ucDigShow8=13; //显示o

ucDigShow7=14; //显示F

ucDigShow6=14; //显示F

}

}


if(ucWd1Part2Update==1) //更新显示当前系统是处于正方向还是反方向

{

ucWd1Part2Update=0; //及时把更新变量清零,防止一直进来更新

if(ucLedDirFlag==0) //正方向,向上,显示n

{

ucDigShow5=16; //显示n

}

else //反方向,向下,显示U

{

ucDigShow5=15; //显示U

}

}


if(ucWd1Part3Update==1) //更新显示当前系统的速度,此数值越大速度越慢,此数值越小速度越快。

{

ucWd1Part3Update=0; //及时把更新变量清零,防止一直进来更新


ucDigShow4=10; //显示空 这一位不用,作为空格


if(uiSetTimeLevel_09_16>=100)

{

ucDigShow3=uiSetTimeLevel_09_16/100; //显示速度的百位

}

else

{

ucDigShow3=10; //显示空

}


if(uiSetTimeLevel_09_16>=10)

{

ucDigShow2=uiSetTimeLevel_09_16%100/10; //显示速度的十位

}

else

{

ucDigShow2=10; //显示空

}


ucDigShow1=uiSetTimeLevel_09_16%10; //显示速度的个位

}






}



void key_scan()//按键扫描函数 放在定时中断里

{


if(key_sr1==1)//IO是高电平,说明按键没有被按下,这时要及时清零一些标志位

{

ucKeyLock1=0; //按键自锁标志清零

uiKeyTimeCnt1=0;//按键去抖动延时计数器清零,此行非常巧妙,是我实战中摸索出来的。

}

else if(ucKeyLock1==0)//有按键按下,且是第一次被按下

{

uiKeyTimeCnt1++; //累加定时中断次数

if(uiKeyTimeCnt1>const_key_time1)

{

uiKeyTimeCnt1=0;

ucKeyLock1=1; //自锁按键置位,避免一直触发

ucKeySec=1; //触发1号键

}

}


if(key_sr2==1)//IO是高电平,说明按键没有被按下,这时要及时清零一些标志位

{

ucKeyLock2=0; //按键自锁标志清零

uiKeyTimeCnt2=0;//按键去抖动延时计数器清零,此行非常巧妙,是我实战中摸索出来的。

}

else if(ucKeyLock2==0)//有按键按下,且是第一次被按下

{

uiKeyTimeCnt2++; //累加定时中断次数

if(uiKeyTimeCnt2>const_key_time2)

{

uiKeyTimeCnt2=0;

ucKeyLock2=1; //自锁按键置位,避免一直触发

ucKeySec=2; //触发2号键

}

}


if(key_sr3==1)//IO是高电平,说明按键没有被按下,这时要及时清零一些标志位

{

ucKeyLock3=0; //按键自锁标志清零

uiKeyTimeCnt3=0;//按键去抖动延时计数器清零,此行非常巧妙,是我实战中摸索出来的。

}

else if(ucKeyLock3==0)//有按键按下,且是第一次被按下

{

uiKeyTimeCnt3++; //累加定时中断次数

if(uiKeyTimeCnt3>const_key_time3)

{

uiKeyTimeCnt3=0;

ucKeyLock3=1; //自锁按键置位,避免一直触发

ucKeySec=3; //触发3号键

}

}


if(key_sr4==1)//IO是高电平,说明按键没有被按下,这时要及时清零一些标志位

{

ucKeyLock4=0; //按键自锁标志清零

uiKeyTimeCnt4=0;//按键去抖动延时计数器清零,此行非常巧妙,是我实战中摸索出来的。

}

else if(ucKeyLock4==0)//有按键按下,且是第一次被按下

{

uiKeyTimeCnt4++; //累加定时中断次数

if(uiKeyTimeCnt4>const_key_time4)

{

uiKeyTimeCnt4=0;

ucKeyLock4=1; //自锁按键置位,避免一直触发

ucKeySec=4; //触发4号键

}

}


}



void key_service() //按键服务的应用程序

{

switch(ucKeySec) //按键服务状态切换

{

case 1:// 改变跑马灯方向的按键 对应学习板的S1键


if(ucLedDirFlag==0) //通过中间变量改变跑马灯的方向

{

ucLedDirFlag=1;

}

else

{

ucLedDirFlag=0;

}


ucWd1Part2Update=1; //及时更新显示方向


uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0; //响应按键服务处理程序后,按键编号清零,避免一致触发

break;


case 2:// 加速按键 对应学习板的S5键 uiSetTimeLevel_09_16越小速度越快

uiSetTimeLevel_09_16=uiSetTimeLevel_09_16-10;

if(uiSetTimeLevel_09_16<50) //最快限定在50

{

uiSetTimeLevel_09_16=50;

}


ucWd1Part3Update=1; //及时更新显示速度


uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0; //响应按键服务处理程序后,按键编号清零,避免一致触发

break;


case 3:// 减速按键 对应学习板的S9键 uiSetTimeLevel_09_16越大速度越慢

uiSetTimeLevel_09_16=uiSetTimeLevel_09_16+10;

if(uiSetTimeLevel_09_16>550) //最慢限定在550

{

uiSetTimeLevel_09_16=550;

}

ucWd1Part3Update=1; //及时更新显示速度

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0; //响应按键服务处理程序后,按键编号清零,避免一致触发

break;


case 4:// 启动和暂停按键 对应学习板的S13键 ucLedStartFlag为0时代表暂停,为1时代表启动


if(ucLedStartFlag==1) //启动和暂停两种状态循环切换

{

ucLedStartFlag=0;

}

else //启动和暂停两种状态循环切换

{

ucLedStartFlag=1;

}

ucWd1Part1Update=1; //及时更新显示系统的运行状态,是运行还是暂停.

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0; //响应按键服务处理程序后,按键编号清零,避免一致触发

break;

}

}





void led_update() //LED更新函数

{


if(ucLed_update==1)

{

ucLed_update=0; //及时清零,让它产生只更新一次的效果,避免一直更新。


if(ucLed_dr1==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x01;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xfe;

}


if(ucLed_dr2==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x02;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xfd;

}


if(ucLed_dr3==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x04;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xfb;

}


if(ucLed_dr4==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x08;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xf7;

}



if(ucLed_dr5==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x10;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xef;

}



if(ucLed_dr6==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x20;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xdf;

}



if(ucLed_dr7==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x40;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0xbf;

}



if(ucLed_dr8==1)

{

ucLedStatus08_01=ucLedStatus08_01|0x80;

}

else

{

ucLedStatus08_01=ucLedStatus08_01&0x7f;

}


if(ucLed_dr9==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x01;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xfe;

}


if(ucLed_dr10==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x02;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xfd;

}


if(ucLed_dr11==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x04;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xfb;

}


if(ucLed_dr12==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x08;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xf7;

}



if(ucLed_dr13==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x10;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xef;

}



if(ucLed_dr14==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x20;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xdf;

}



if(ucLed_dr15==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x40;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0xbf;

}



if(ucLed_dr16==1)

{

ucLedStatus16_09=ucLedStatus16_09|0x80;

}

else

{

ucLedStatus16_09=ucLedStatus16_09&0x7f;

}


hc595_drive(ucLedStatus16_09,ucLedStatus08_01); //74HC595底层驱动函数


}

}



void display_drive()

{

//以下程序,如果加一些数组和移位的元素,还可以压缩容量。但是鸿哥追求的不是容量,而是清晰的讲解思路

switch(ucDisplayDriveStep)

{

case 1: //显示第1位

ucDigShowTemp=dig_table[ucDigShow1];

if(ucDigDot1==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfe);

break;

case 2: //显示第2位

ucDigShowTemp=dig_table[ucDigShow2];

if(ucDigDot2==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfd);

break;

case 3: //显示第3位

ucDigShowTemp=dig_table[ucDigShow3];

if(ucDigDot3==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfb);

推荐阅读

史海拾趣

Elpaq公司的发展小趣事
爱特姆(ATOM)公司的发展小趣事

爱特姆深知人才是企业发展的核心动力。因此,公司一直注重人才的引进和培养。通过招聘行业内的优秀人才,建立了一支高素质、专业化的研发团队。同时,公司还定期举办各类培训活动,提升员工的技能水平和综合素质。

CYANLITE公司的发展小趣事

在XXXX年,CYANLITE公司迎来了一次重要的技术突破。他们成功研发出了一种新型的LED芯片,这种芯片具有更高的发光效率和更长的使用寿命。这一技术的突破使得CYANLITE公司的产品在市场上更具竞争力,公司也借此机会进一步扩大了市场份额。随着品牌知名度的提升,CYANLITE公司开始拓展国际市场,其产品逐渐出口到欧洲、北美等地区,公司的业绩也实现了快速增长。

Garmin_Canada_Inc.公司的发展小趣事
考虑到实际应用环境的复杂性,拨号电路应具有一定的环境适应性。例如,在电磁干扰较强的环境中,采取适当的屏蔽和滤波措施;在温度变化较大的环境中,采用温度补偿电路等。
ATP [ATP Electronics]公司的发展小趣事

面对工业应用领域的特殊需求,ATP敏锐地捕捉到了市场机遇。公司成功推出了业界首款工业级SLC e.MMC产品E800Pi,这款产品以其高达6万次的单元擦写耐久性和出色的耐高温、交叉温度能力,受到了工业领域的广泛认可。E800Pi的推出,不仅拓宽了ATP的产品线,也进一步提升了公司在工业存储解决方案领域的市场地位。

Alcatel-Lucent公司的发展小趣事

为了适应全球市场的变化,ATP积极推进全球化战略。公司在美国硅谷设立总部,并在英国、台湾、中国大陆等地设立分公司,构建起了全球化的研发、生产和销售网络。这一战略布局不仅增强了ATP在全球市场的竞争力,也为公司带来了更多的合作机会和发展空间。

问答坊 | AI 解惑

基于89c52的公交车ic卡设计

基于89c52的公交车ic卡设计 公交车IC卡电路图,原程序,以前收集的资料,现在共享给大家希望共同进步。 …

查看全部问答>

mini2440在开机或者复位的时候按下开发板上的 K1-K6任意按键,请教问题

请教! 呵呵,搞了半天 没反应 mini2440在开机或者复位的时候按下开发板上的 K1-K6任意按键,这样我们就进入了vivi 模式 可是我按了半天还是出不来,大家有人做过的吧?教教我呢,谢谢了!…

查看全部问答>

关于新建BSP的问题

现在我手头有块SMDK2442的开发板和BSP,一切都可以正常工作;我又重新做了一个开发板,也是用2442的,比方说,就叫做MV2442吧,我想新建一个BSP,但我又不想破坏之前的SMDK2442的BSP,就是说,我想克隆一个BSp,在我的工程下有两个BSP,当编译SMDK2 ...…

查看全部问答>

请问高手,UI(用户接口)和驱动程序交互

最近在做一个打印用户接口插件,主要是想控制打印驱动,在打印时使用的分辨率(DPI),打印的色彩, 以及输出路径(一个虚拟打印机)等信息。 请问如何把这些参数传递给打印驱动程序呢?     希望能把整个过程描述一下。谢谢!!…

查看全部问答>

【请教】msp430的spi的如何清空Data shift register

麻烦大家了,问了好多关于这个的问题 我用dsp与msp430通信,发生数据位偏移现象 我发现很有可能是由于在一开始dsp前期的一些操作中(还没有开始spi通讯时),会产生2个CLK时钟,让Data shift register预先的偏移2位,开始spi通讯以后,每个数据都 ...…

查看全部问答>

求一个含有电能计量芯片元件的proteus仿真元件库

我的元件库里找不到电能芯片,又不会自己做,只好厚颜来这里求成品了。 只要那个元件库有ADE7755或者ADE7757或者CS5460A其中一个就行了,请好心人发到我邮箱吧baimua@163.com。感谢!!!!…

查看全部问答>

【一点一点学Linux C】Kmalloc

  kmalloc/kfree,vmalloc/vfree函数用法和区别1.kmalloc   kmalloc内存分配和malloc相似,除非被阻塞否则他执行的速度非常快,而且不对获得空间清零.说明:在用kmalloc申请函数后,要清零用memset()函数对申请的内存进行清零。 ...…

查看全部问答>

SPI Bootloader for Hercules TMS570LS04x MCU

SPI Bootloader for Hercules TMS570LS04x MCU.pdf…

查看全部问答>

论坛发帖普通模式到高级模式Bug?

今天本来准备发LPC4357开发第一篇调试的,但是刚刚就在普通的发帖模式下写的字,然后想上图片,发现这个这个模式上图片添加不了,于是点击了高级模式,到转过去后所有写的字都没有了,郁闷啊。 不知道对于找个是否能改进,转高级模式原来编辑的文 ...…

查看全部问答>