单片机定时器实验两位倒计时秒表_Proteus入门单片机(3)例程分析

0d155a8a8a746be888d61c9b49e9f2b5.png

74HC573详解:

7075de5d65c05ae59e529f604f614fc5.png
/*
调试环境:Proteus 7.8  AT89S52单片机  8位共阴数码管
功能为:实现数码管1秒~99秒倒计时,    12M晶振
*/

#include <reg52.h>
#define uchar unsigned char
sbit LED1=P1^1;                           //连的是LED1
#define MAX_NUM     13                //最大倒计时数1~99可设
#define SEGPORT         P0           //定义数码管连接端口
sbit LE_DUAN         = P2^0;           //定义573锁存使能端口 段锁存
sbit LE_WEI         = P2^1;           //定义573锁存使能端口 位锁存

unsigned char code Seg_Duan[] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};//共阴数码管显示段码值 0~9
unsigned char code Seg_Wei[]  = {0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};                 //数码管的位码,低电平有效。

static uchar second=MAX_NUM,count;    //99只是一个数,可以任意改,因为这里只学习怎样实现倒计时

uchar shiwei;
uchar gewei;

void delay(unsigned int cnt)
{
        while(--cnt);
}

void main()
{
        TMOD|=0x01;       //定时器设置 16位
        TH0=(65536-50000)/256;  //定时时间   50ms
        TL0=(65536-50000)%256;
        EA=1;
        ET0=1;
        TR0=1;
        SEGPORT = 0;//P0等于0,段码位码均为零
        LE_DUAN = 0;//74HC573锁
        LE_WEI = 0;//74HC573锁
        while(1)
        {
                shiwei=second/10;  //数码管10位
                gewei=second%10;   //数码管个位
                       //下面四句送段码,
                SEGPORT=0;    //清空数据,防止有交替重影
                LE_DUAN=1;      //段锁存
                LE_DUAN=0;
                delay(30);

                SEGPORT = Seg_Wei[1];     //先送位码值
                LE_WEI = 1;           //573高电平锁存
                LE_WEI = 0;
                SEGPORT = Seg_Duan[shiwei];    //再送段码值,这里为 2~9
                LE_DUAN = 1;                //573高电平锁存
                LE_DUAN = 0;
                delay(300);


                SEGPORT = Seg_Wei[0];           //先送位码值
                LE_WEI = 1;           //573高电平锁存
                LE_WEI = 0;
                SEGPORT = Seg_Duan[gewei]; //再送段码值,这里为 2~9
                LE_DUAN = 1;                //573高电平锁存
                LE_DUAN = 0;
                delay(300);
        }
}

void tim(void) interrupt 1    //定时器0中断    
{
        TH0=(65536-50000)/256;     //定时时间   50ms
        TL0=(65536-50000)%256;
        count++;
    if(count==20)     //50毫秒定时,50*20=1000(毫秒)=1秒
    {
       count=0;
       second--;
       if(second==0xff)/*无符号char型数值,-1为:0xff,0之后,显示最大倒计数。*/
       {
         LED1=~LED1;      //这里让LED1反转一下,表现倒计时一周。
         second=MAX_NUM;   //回到最大倒计数值,再次倒计,当然,可以加入其他的控制
       }
         
    }
}
————————————————
版权声明:本文为CSDN博主「tcjy1000」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/tcjy1000/article/details/19331919

分析:在死循环里,一直重复着以下三个操作

第一:分离second的十位和个位,消隐

第二:十位送到十位

第三:个位送到个位

什么导致second的变化?

答:通过定时器与中断,在定时时间到在中断子程序中改变计数值。

实验有什么问题?

首先硬件连接了8颗LED,这里只用了一颗,动态数码管也没有发挥他的功能,然后在进中断时程序的位置无法确定,不能做到精准倒计时。

如何改进?

使用38译码器可以进一步节省代码,将led改为静态数码管可以记录倒计时次数。

学到了什么?

74HC573可以用来控制动态数码管,38译码器可以节省引脚。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值