注意"引脚号",不同的单片机,接入的I/O口可能不同。代码依据下方给出的仿真图接线进行编写。
/**************************
Title :数码管动态显示(定时器+中断)
Author:Guanglei Bie
E-mail:bglei@foxmail.com
Data: 2023/6/6
**************************/
#include "reg52.h"
#include "intrins.h"
sbit LSA=P3^0; //38译码器的输入
sbit LSB=P3^1;
sbit LSC=P3^2;
#define led P2
unsigned char flag1s=0;
unsigned char code smgduan[16]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; //共阴极数码管显示0~F的值,code表示存储在flash区,给RAM节省内存
unsigned char ledbuff[8]={0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};
unsigned char i=0;
unsigned int cnt=0;
unsigned long sec=0;
void main()
{
LSA=0;
LSB=0;
LSC=0;
TMOD=0x01; //定时器模式1
TH0=0xFC; //定时器16位初值0xFC67定时大约1ms
TL0=0x67; //“100hz无闪烁”:也就是10ms内的变化人眼很难看出闪烁,1ms一位,8位数码管共8ms,符合要求
TR0=1; //定时器启动
EA=1; //中断使能
ET0=1; //中断0启动
led=0x7f;
while(1)
{
if(flag1s==1)
{
flag1s=0;
led=_crol_(led,1);
sec++;
if(sec==100000000)
{
sec=0;
}
ledbuff[7]=smgduan[sec%10];
ledbuff[6]=smgduan[sec/10%10];
ledbuff[5]=smgduan[sec/100%10];
ledbuff[4]=smgduan[sec/1000%10];
ledbuff[3]=smgduan[sec/10000%10];
ledbuff[2]=smgduan[sec/100000%10];
ledbuff[1]=smgduan[sec/1000000%10];
ledbuff[0]=smgduan[sec/10000000%10];
}
}
}
void interruptTime0() interrupt 1 //定时器溢出时自动进入中断,中断自动给定时器溢出位清零
{
TH0=0xfc;
TL0=0x67;
cnt++;
if(cnt>=1000)
{
cnt=0;
flag1s=1;
}
P0=0x00; //有些段闪微光(鬼影),消除数码管的鬼影(消影)
switch(i)
{
case 0:LSC=0;LSB=0;LSA=0;i++;P0=ledbuff[0];break;
case 1:LSC=0;LSB=0;LSA=1;i++;P0=ledbuff[1];break;
case 2:LSC=0;LSB=1;LSA=0;i++;P0=ledbuff[2];break;
case 3:LSC=0;LSB=1;LSA=1;i++;P0=ledbuff[3];break;
case 4:LSC=1;LSB=0;LSA=0;i++;P0=ledbuff[4];break;
case 5:LSC=1;LSB=0;LSA=1;i++;P0=ledbuff[5];break;
case 6:LSC=1;LSB=1;LSA=0;i++;P0=ledbuff[6];break;
case 7:LSC=1;LSB=1;LSA=1;i=0;P0=ledbuff[7];break;
default:break;
}
}
//附1:1ms(0.001s)定时初值设定 x*12/11059200=0.001解得x=921.6 65536-921=64615对应的十六进制为0xFC67
//附2:中断的作用是保证计算过程(需要时间)不影响数码管的赋值(避免赋值时间不均匀导致的闪烁),只要达到1ms,不管main函数进行到哪儿,立马进入中断进行赋值,赋值时间均匀
//注: 欢迎邮箱交流,见文尾
数码管动态显示(定时器+中断)
欢迎一起交流学习,邮箱:bglei@foxmail.com