一个在一小时以内实现分钟可调的倒计时闹钟

本文介绍了一个基于51单片机的倒计时程序实现细节,包括定时器配置、数码管动态显示、按键控制等功能。通过具体代码展示了如何设置定时器0的工作模式、初始化参数以及中断处理等关键步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

#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;
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值