金沙滩51单片机数码管的动态显示

我们的目标是实现秒表功能,只不过这次有 6 个位了,最大可以计到 999999 秒。

在这里插入图片描述

#include <reg52.h>

sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;

unsigned char code LedChar[] = {  //数码管显示字符转换表
    0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
    0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
unsigned char LedBuff[6] = {  //数码管显示缓冲区,初值0xFF确保启动时都不亮
    0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};

void main()
{
    unsigned char i = 0;    //动态扫描的索引
    unsigned int  cnt = 0;  //记录T0中断次数
    unsigned long sec = 0;  //记录经过的秒数

    ENLED = 0;    //使能U3,选择控制数码管
    ADDR3 = 1;    //因为需要动态改变ADDR0-2的值,所以不需要再初始化了
    TMOD = 0x01;  //设置T0为模式1
    TH0  = 0xFC;  //为T0赋初值0xFC67,定时1ms
    TL0  = 0x67;
    TR0  = 1;     //启动T0
    
    while (1)
    {
        if (TF0 == 1)         //判断T0是否溢出
        {
            TF0 = 0;          //T0溢出后,清零中断标志
            TH0 = 0xFC;       //并重新赋初值
            TL0 = 0x67;
            cnt++;            //计数值自加1
            if (cnt >= 1000)  //判断T0溢出是否达到1000次
            {
                cnt = 0;      //达到1000次后计数值清零
                sec++;        //秒计数自加1
                //以下代码将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];
            }
            //以下代码完成数码管动态扫描刷新
            switch (i)
            {
                case 0: ADDR2=0; ADDR1=0; ADDR0=0; i++; P0=LedBuff[0]; break;
                case 1: ADDR2=0; ADDR1=0; ADDR0=1; i++; P0=LedBuff[1]; break;
                case 2: ADDR2=0; ADDR1=1; ADDR0=0; i++; P0=LedBuff[2]; break;
                case 3: ADDR2=0; ADDR1=1; ADDR0=1; i++; P0=LedBuff[3]; break;
                case 4: ADDR2=1; ADDR1=0; ADDR0=0; i++; P0=LedBuff[4]; break;
                case 5: ADDR2=1; ADDR1=0; ADDR0=1; i=0; P0=LedBuff[5]; break;
                default: break;
            }
        }
    }
}

问题

  • 数码管的不应该亮的段,似乎有微微的发亮,这种现象叫做“鬼影”。
    关闭段:在 switch(i)这句程序之前,加一句 P0=0xFF;这样就把数码管所有的段都关闭了,
    当把“ADDR”的值全部搞定后,再给 P0 赋对应的值即可。
    关闭位:在 switch(i)这句程序之前,加上一句 ENLED=1;等到把 ADDR2=0; ADDR1=0;
    ADDR0=0; i++; P0=LedBuff[0];这几条刷新程序全部写完后,再加上一句 ENLED=0;然后再进行 break 操作即可。

  • 我们的数码管上的数字每一秒变化一次,变化的时候,不参加变化的数码管可能出现一次抖动。

最终版,使用中断

#include <reg52.h>

sbit ADDR0 = P1^0;
sbit ADDR1 = P1^1;
sbit ADDR2 = P1^2;
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;

unsigned char code LedChar[] = {  //数码管显示字符转换表
    0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
    0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
unsigned char LedBuff[6] = {  //数码管显示缓冲区,初值0xFF确保启动时都不亮
    0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};
unsigned char i = 0;   //动态扫描的索引
unsigned int cnt = 0;  //记录T0中断次数
unsigned char flag1s = 0;  //1秒定时标志

void main()
{
    unsigned long sec = 0;  //记录经过的秒数

    EA = 1;       //使能总中断
    ENLED = 0;    //使能U3,选择控制数码管
    ADDR3 = 1;    //因为需要动态改变ADDR0-2的值,所以不需要再初始化了
    TMOD = 0x01;  //设置T0为模式1
    TH0  = 0xFC;  //为T0赋初值0xFC67,定时1ms
    TL0  = 0x67;
    ET0  = 1;     //使能T0中断
    TR0  = 1;     //启动T0
    
    while (1)
    {
        if (flag1s == 1)  //判断1秒定时标志
        {
            flag1s = 0;   //1秒定时标志清零
            sec++;        //秒计数自加1
            //以下代码将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];
        }
    }
}
/* 定时器0中断服务函数 */
void InterruptTimer0() interrupt 1
{
    TH0 = 0xFC;  //重新加载初值
    TL0 = 0x67;
    cnt++;       //中断次数计数值加1
    if (cnt >= 1000)  //中断1000次即1秒
    {
        cnt = 0;      //清零计数值以重新开始下1秒计时
        flag1s = 1;   //设置1秒定时标志为1
    }
    //以下代码完成数码管动态扫描刷新
    P0 = 0xFF;   //显示消隐
    switch (i)
    {
        case 0: ADDR2=0; ADDR1=0; ADDR0=0; i++; P0=LedBuff[0]; break;
        case 1: ADDR2=0; ADDR1=0; ADDR0=1; i++; P0=LedBuff[1]; break;
        case 2: ADDR2=0; ADDR1=1; ADDR0=0; i++; P0=LedBuff[2]; break;
        case 3: ADDR2=0; ADDR1=1; ADDR0=1; i++; P0=LedBuff[3]; break;
        case 4: ADDR2=1; ADDR1=0; ADDR0=0; i++; P0=LedBuff[4]; break;
        case 5: ADDR2=1; ADDR1=0; ADDR0=1; i=0; P0=LedBuff[5]; break;
        default: break;
    }
}

单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考 单片机 学习资料 项目资料 供学习参考
实现该功能需要以下步骤: 1. 接线:将数码管的阳极接到单片机的P0口,将8个阴极接到P1~P8口。 2. 初始化:设置P0口为输出,P1~P8口为输入,同时设置定时器1的计时器初值为65536-12000,即每1ms中断一次。 3. 倒计时:在定时器1中断函数中,每1ms更新计时器计数值,同时判断是否到达1秒,如果到达1秒,则将秒数减1,并更新数码管显示。 4. 显示:将秒数转换为BCD码,分别存储在P1~P4口中,然后依次输出到数码管。 以下是代码实现: ``` #include <reg51.h> #define uint unsigned int #define uchar unsigned char sbit dula = P2^6; sbit wela = P2^7; uchar code table[]={ 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f }; uint count = 60000; // 倒计时60秒 void delay_ms(uint xms) // 延时函数 { uint i,j; for(i = xms;i > 0;i--) for(j = 110;j > 0;j--); } void display(uchar num) // 显示函数 { P1 = table[num/1000]; P2 = 0xfe; delay_ms(1); P2 = 0xff; P1 = table[num%1000/100]; P2 = 0xfd; delay_ms(1); P2 = 0xff; P1 = table[num%100/10]; P2 = 0xfb; delay_ms(1); P2 = 0xff; P1 = table[num%10]; P2 = 0xf7; delay_ms(1); P2 = 0xff; } void timer1() interrupt 3 // 定时器1中断函数 { TH1 = (65536-12000)/256; TL1 = (65536-12000)%256; count--; if(count == 0) // 到达1秒,减1秒 { count = 60000; display(count/1000); // 显示秒数 } } void main() { TMOD = 0x10; // 定时器1工作在方式1 TH1 = (65536-12000)/256; TL1 = (65536-12000)%256; ET1 = 1; // 打开定时器1中断 EA = 1; // 打开总中断 TR1 = 1; // 启动定时器1 P0 = 0x00; P1 = 0x00; P2 = 0xff; P3 = 0xff; P4 = 0xff; while(1) { dula = 1; P0 = 0x00; dula = 0; wela = 1; P0 = 0x7f; wela = 0; } } ``` 需要注意的是,这里使用了定时器1中断来实现每1秒减1秒的倒计时,同时更新数码管显示。数码管的显示函数中,使用了延时函数来控制显示时间。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值