74HC573详解:
/*
调试环境:Proteus 7.8 AT89S52单片机 8位共阴数码管
功能为:实现数码管1秒~99秒倒计时, 12M晶振
*/
#include
#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; //回到最大倒计数值,再次倒计,当然,可以加入其他的控制
}
}
}
分析:在死循环里,一直重复着以下三个操作
第一:分离second的十位和个位,消隐
第二:十位送到十位
第三:个位送到个位
什么导致second的变化?
答:通过定时器与中断,在定时时间到在中断子程序中改变计数值。
实验有什么问题?
首先硬件连接了8颗LED,这里只用了一颗,动态数码管也没有发挥他的功能,然后在进中断时程序的位置无法确定,不能做到精准倒计时。
如何改进?
使用38译码器可以进一步节省代码,将led改为静态数码管可以记录倒计时次数。
学到了什么?
74HC573可以用来控制动态数码管,38译码器可以节省引脚。