#include<reg52.h>
#define uchar unsigned char
#define ul unsigned long
sbit d1=P1^0;
sbit dula=P2^6;//声明锁存器U1的锁存端
sbit wela=P2^7;//声明锁存器U2的锁存端
sbit beep=P2^3;
sbit key1=P3^4;
sbit key2=P3^5;
sbit key3=P3^6;
sbit key4=P3^7;
uchar code table[]=
{
0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71
};
uchar code tablwe[]=
{
0xdf,0xef,0xf7,0xfb,0xfd,0xfe
};
int count,min10,min,s10,s,ms100,ms10,shuzi1,shuzi2,wei,num,judge;
void delayms(int);
void wedu(int,int);
void display();
void keyscan();
void initial()
{
TMOD=0x01; //定时器0工作于方式1
TH0=(65536-4608)/256;//装初值
TL0=(65536-4608)%256;
EA=1;
ET0=1;
TR0=1;
min10=5;
min=9;
s10=5;
s=9;
ms100=9;
ms10=9;
}
void main()
{
initial();
while(1)//程序在此处不停地对数码管进行动态扫描等待中断发生数码管示数发生变化
{
keyscan();
display();
}
}
void display()
{
for(wei=0;wei<6;wei++)
{
switch(wei)
{
case 0: wedu(ms10,wei);break;
case 1: wedu(ms100,wei);break;
case 2: wedu(s,wei);break;
case 3: wedu(s10,wei);break;
case 4: wedu(min,wei);break;
case 5: wedu(min10,wei);break;
}
}
}
void delayms(int xms)
{
int i,j;
for(i=xms;i>0;i--)
for(j=1;j>0;j--);
}
void wedu(int shuzi1,int shuzi2)
{
dula=1;
P0=table[shuzi1];//送入段选数据
dula=0;
P0=0xff;//送位选数据前关闭所有显示,防止打开位选锁存器时原来段选数据通过位选锁存器造成混乱
wela=1;
P0=tablwe[shuzi2];//进一步简化
wela=0;
P0=0xff;//送位选数据前关闭所有显示,防止打开位选锁存器时原来段选数据通过位选锁存器造成混乱
delayms(150);
}
void T0_time() interrupt 1
{
TH0=(65536-4608)/256;//重装初值
TL0=(65536-4608)%256;
num++;
if(num==2)
{
num=0;
ms10--;
if(ms10==-1)
{
ms10=9;
ms100--;
}
if(ms100==-1)
{
ms100=9;
s--;
beep=0;
delayms(5);
beep=1;
delayms(5);
}
if(s==-1)
{
s=9;
s10--;
}
if(s10==-1)
{
s10=5;
min--;
}
if(min==-1)
{
min=9;
min10--;
}
if(min10==-1)
{
min10=0;
min=0;
s10=0;
s=0;
ms100=0;
ms10=0;
TR0=0;
judge=1;
while(judge)
{
display();
min10=0;
min=0;
s10=0;
s=0;
ms100=0;
ms10=0;
TR0=0;
beep=0;
delayms(5000);
beep=1;
delayms(5000);
if((key3==0)||(key4==0))
{
delayms(10);//检测按下延迟
if((key3==0)||(key4==0))
{
TR0=0;
d1=0;
beep=0;
min10=5;
min=9;
s10=5;
s=9;
ms100=9;
ms10=9;
judge=0;
}
while((!key3)||(!key4));//松手检测,不都松开,while始终循环
//都松开,while退出,退出此次检测程序
delayms(5);
while((!key3)||(!key4));//检测松手延迟
}
else
{
d1=1;
beep=1;
}
}
}
}
}
void keyscan()
{
if((key1==0)&&(TR0==0))
{
delayms(10);//检测按下延迟
if((key1==0)&&(TR0==0))
{
d1=0;
beep=0;
min++;
if(min==10)
{
min=0;
min10++;
if(min10==6) min10=0;
}
}
while(!key1);//松手检测,不松开key1=0,while始终循环
//松开key1=1,while退出,退出此次检测程序
delayms(5);
while(!key1);//检测松手延迟
}
else
{
d1=1;
beep=1;
}
if((key2==0)&&(TR0==0))
{
delayms(10);//检测按下延迟
if((key2==0)&&(TR0==0))
{
d1=0;
beep=0;
min--;
if(min==-1)
{
min=9;
min10--;
if(min10==-1) min10=5;
}
}
while(!key2);//松手检测,不松开key2=0,while始终循环
//松开key2=1,while退出,退出此次检测程序
delayms(5);
while(!key2);//检测松手延迟
}
else
{
d1=1;
beep=1;
}
if(key3==0)
{
delayms(10);//检测按下延迟
if(key3==0)
{
TR0=0;
d1=0;
beep=0;
min10=5;
min=9;
s10=5;
s=9;
ms100=9;
ms10=9;
}
while(!key3);//松手检测,不松开key3=0,while始终循环
//松开key3=1,while退出,退出此次检测程序
delayms(5);
while(!key3);//检测松手延迟
}
else
{
d1=1;
beep=1;
}
if(key4==0)
{
delayms(10);//检测按下延迟
if(key4==0)
{
d1=0;
beep=0;
TR0=~TR0;
}
while(!key4);//松手检测,不松开key4=0,while始终循环
//松开key4=1,while退出,退出此次检测程序
delayms(5);
while(!key4);//检测松手延迟
}
else
{
d1=1;
beep=1;
}
if((key1==0)&&(key2==0))
{
delayms(10);//检测按下延迟
if((key1==0)&&(key2==0))
{
TR0=1;
d1=0;
beep=0;
min10=0;
min=0;
s10=0;
s=0;
ms100=0;
ms10=0;
}
while((!key1)&&(!key2));//松手检测,都不松开key3=0,while始终循环
//只要松开key1或者key2,while退出,退出此次检测程序
delayms(5);
while((!key1)&&(!key2));//检测松手延迟
}
else
{
d1=1;
beep=1;
}
}
一个在一小时以内实现分钟可调的倒计时闹钟
最新推荐文章于 2023-10-12 08:47:59 发布