数码管动态显示,(中断)

先上代码 芯片为STC89C52RC

#include <REGX52.H>

unsigned char i=0;//某位数码管
unsigned int msec=0;
unsigned long sec=0;

sbit P22=P2^2;
sbit P23=P2^3;
sbit P24=P2^4;

//数码管显示内容,共阴数码管真值表0~9,A~F
unsigned char code LedChar[16]={
0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,
0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71};


unsigned char LedBuff[8]={0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};//数码管缓冲区,16进制


void Led(unsigned char a)//激活第a个数码管,a介于0~7
{
    switch(a)
    {
        case 0: P22=0;P23=0;P24=0;break;
        case 1: P22=1;P23=0;P24=0;break;
        case 2: P22=0;P23=1;P24=0;break;
        case 3: P22=1;P23=1;P24=0;break;
        case 4: P22=0;P23=0;P24=1;break;
        case 5: P22=1;P23=0;P24=1;break;
        case 6: P22=0;P23=1;P24=1;break;
        case 7: P22=1;P23=1;P24=1;break;
    }
}

void main()
{
    signed char b;
    TMOD=0x01;//定时器模式
    TH0=0xFC;
    TL0=0x67;//定时1ms
    TR0=1;//启动定时器
    EA=1;//打开总中断
    ET0=1;//定时器0中断使能
    while(1)
    {
        if(msec>=1000)//每秒刷新缓冲区
        {
            msec=0;
            sec++;
            LedBuff[0]=LedChar[sec%10];
            LedBuff[1]=LedChar[(sec/10)%10];
            LedBuff[2]=LedChar[(sec/100)%10];
            LedBuff[3]=LedChar[(sec/1000)%10];
            LedBuff[4]=LedChar[(sec/10000)%10];
            LedBuff[5]=LedChar[(sec/100000)%10];
            LedBuff[6]=LedChar[(sec/1000000)%10];
            LedBuff[7]=LedChar[(sec/10000000)%10];
            for(b=7;b>=0;b--)//高位消零
            {
                if(LedBuff[b]==0x3f)
                {
                    LedBuff[b]=0x00;
                }
                else break;
            }
        }
    }
}

void InterruptTimer0() interrupt 1    //中断入口号*8+3=中断向量地址,溢出就触发中断
{
    msec++;
    TH0=0xFC;
    TL0=0x67;
    P0=0x00;//消影
    Led(i);
    P0=LedBuff[i];
    i++;
    if(i==8)i=0;
}

防止闪烁

计算LedBuff[]所消耗时间不可忽略,再延时后可能刷新每位数码管时间不均,造成闪烁,

就把数码管的激活、显示程序放在中断函数里,保证每位数码管显示时间都为1ms。

消影:若未使用消影,可能使本该灭的某个LED产生微弱的亮度

原因为:假设将要激活第四位数码管时,第28行 执行完P22=0,23=0;,还未执行P24=1;时,

此时状态为P22,P23,P24全给低电平(38译码器),激活了第0位数码管

而P0继续给LedBuff[3],就在短暂的时间内把第三位数码管显示的内容错显示到了第0位数码管上

方法为在中断程序中将要激活和显示数码管之前,P0给0,即全灭,激活某位数码管完成后再P0给LedBuff[],避免了显示错位

效果:下面为消影后

高位消零。(防止显示“100”时将后两位0也消掉)

思路是: 从最高位开始检测缓冲区的元素,检测到0x3F就替换为0x00,若检测到不是0x3F就不检测后面的,跳出循环。

(0x3F是共阴数码管的0,0x00是不显示)

单看for循环的话b是有可能为-1的,我第一次写就顺手写成unsigned char型了(虽然能正常运行)已改为unsigned char

一切完成后发现,当数码管只有个位时(前9秒),每秒刷新时十位都会闪烁一下,检查后认为问题出现在消零上

猜测原因:消零的for循环占用时间太多,可能某位LedBuff[]算出0后还没来得及消零,或者还没有消到那一位,中断就触发了就显示在数码管上了。

也尝试让消零程序占的时间尽量少,b的初值给2,只消前三位,结果仍然是第二位在每秒刷新的时候有闪烁,消不彻底

该问题还未解决

这是我的思路,或者大佬们有更好的消零方法

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Poetry-

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值