使用STC89C52灵活控制数码管

本文详细介绍了数码管的工作原理,包括其结构和信号传输过程,重点讲解了为何需要74HC245进行信号缓冲。文章还展示了如何通过编程实现动态数码管显示,包括消除因快速切换导致的残影问题。
摘要由CSDN通过智能技术生成

静态数码管显示

在使用数码管之前,我们首先要对数码管的工作原理有一个了解,请查看板子的数码管模块原理图

HYM67SZ28VWR28MQKT126ffac5dc37f9dc.png

可以看到,一共有8位数字,每位数字由8个小LED构成(a、b、c、d、e、f、g、dp)每个数字的对应LED都连在一起,并且与74HC245相连,74HC245的作用是进行缓冲,当DIR端口接高电平时固定为A0传输给B0,A1给B1...

为什么不直接相连而要进行缓冲呢?

因为如果直接相连的话,高电平从寄存器P0传来的信号已经很小,很难再驱动这么多信号,所以此时借用DIR端的高电平,只需要检测到P0传来的对应的信号就可以使用芯片使Vcc的电信号传输给8个LED端口,因此寄存器P0就可以实现操控LED显示的数字。例如:当想要显示数字8时,只需要使a、b、c、d、e、f、g都为亮,也就是接通高电平,此时P0应赋值为0111 1111即0x7F,一位数赋值情况如下表:

0:0011 1111即0x3F1:0000 0110即0x06
2:0101 1011即0x5B3:0100 1111即0x4F
4:0110 0110即0x665:0110 1101即0x6D
6:0111 1101即0x7D7:0000 0111即0x07
8:0111 1111即0x7F9:0110 1111即0x6F

74HC1387e7651229062eb7c.png

如何选择哪一位来显示数字呢?通过查看端口LED0~7,发现他们都与译码器74HC138相连通过寄存器P2中的第2到4为进行选择,当P24为1,当P23为0,当P24为0时,选中Y4即LED5进行显示,此时LED5端口为低电平,所以可以解释为什么P0中的值为1时小LED亮。

举例实现一个使LED2显示6的代码:

#include <REGX52.H>
​
void main()
{
    //想要使LED2亮,就要选中Y1,所以使P2的第四位到第二位分别为001(二进制高位对应寄存器高位)
    P2_2=1;
    P2_3=0;
    P2_4=0;
    P0=0x7D;//显示6,dp和b端为0不亮,其余为1亮 对应0111 1101
    while(1){}
}

这样使用数码管未免有些过于麻烦,为了方便后续的数码管使用,我们可以将LED显示封装为一个函数,只需要给这个函数两个信息,一个是显示的位置,一个是显示的值,这样就可以便捷的使用数码管。实现如下:

#include <REGX52.H>
​
unsigned char Nixienum[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};
//将要显示的每一个数的八个小LED熄灭情况转换为十六进制表示,并与数组下标对应起来
void Nixie(unsigned char Location,num)
{
    switch(Location)//根据传递过来的Location来选择对应位置的LED显示
    {
        case 1:P2_4=1;P2_3=1;P2_2=1;break;
        case 2:P2_4=1;P2_3=1;P2_2=0;break;
        case 3:P2_4=1;P2_3=0;P2_2=1;break;
        case 4:P2_4=1;P2_3=0;P2_2=0;break;
        case 5:P2_4=0;P2_3=1;P2_2=1;break;
        case 6:P2_4=0;P2_3=1;P2_2=0;break;
        case 7:P2_4=0;P2_3=0;P2_2=1;break;
        case 8:P2_4=0;P2_3=0;P2_2=0;break;
    }
    P0=Nixienum[num];//显示对应数字
}
​
void main()
{
    Nixie(2,6);             //只需要传递一个位置和想要显示的数即可
    while(1)
    {
        
    }
}

动态数码管显示

动态数码管显示就是多个LED同时显示不同的数字,但是我们的数码管显示函数只能实现一位数字的显示,但是,左脚踩右脚,一定能上天。只要我们显示的够快,每位数显示的时候,只需要在它消失之前,再次显示一遍,它就能永远留在上面。

因此我们把想要显示的不同位置的不同的数字放入循环体内,使它一直不断地刷新,就可以实现不同的数字同时留在上面。

但是此时我们会发现,数字居然有些错位了!为什么会这样?

举个例子,当我上前一位数的LED灯的a是亮的,但是下一位的a是灭的,我还没来得及熄,你就已经换了下一位LED,所以就会留下一点残影。

如何消除残影呢?

只需要在函数后面加一个延迟一毫秒的Delay函数,这样每次显示完一位成后都会停止一会儿,这样就可以消除残影了。

总体实现如下:

#include <REGX52.H>
​
unsigned char Nixienum[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};
​
void Delay(int n)   //@12.000MHz        ms
{       
    unsigned char data i, j;
    while(n--)
    {
        i = 2;
        j = 239;
        do
        {
            while (--j);
        } while (--i);
    }
}
​
void Nixie(unsigned char Location,num)
{
    switch(Location)
    {
        case 1:P2_4=1;P2_3=1;P2_2=1;break;
        case 2:P2_4=1;P2_3=1;P2_2=0;break;
        case 3:P2_4=1;P2_3=0;P2_2=1;break;
        case 4:P2_4=1;P2_3=0;P2_2=0;break;
        case 5:P2_4=0;P2_3=1;P2_2=1;break;
        case 6:P2_4=0;P2_3=1;P2_2=0;break;
        case 7:P2_4=0;P2_3=0;P2_2=1;break;
        case 8:P2_4=0;P2_3=0;P2_2=0;break;
    }
    P0=Nixienum[num];
    Delay(1);               //消影
}
​
void main()
{
    while(1)
    {
        Nixie(1,1); 
        Nixie(2,2);
        Nixie(3,3);
        Nixie(4,4);
        
    }
}
  • 25
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值