51单片机七段数码管显示时钟加按键控制—③—74HC595版

前期链接:
51单片机七段数码管显示时钟无按键控制—①—74HC595版
51单片机七段数码管显示时钟加按键控制—②—74HC595版

一、简要说明:

1、时钟第一版只显示了最基本的时钟数值,没有加入时钟设置。
2、第二版加入了通过按键设置时间的数值(button123)和切换显示 时间/日期(button4)
3、第三版也就是这一版加入了按键设置日期,并且实现在设置时间/日期 相应设置位的闪烁。

二、遇到的问题:

在设置位数闪烁时,由于我想采用定时器计数,但我发现我在之前的代码中进入设置状态后会把定时器关闭,无法通过计时器计数

当时想到的解决方案有三
①另外添加一个定时器——我懒,不想再加
②不采用准确计数,而采用延时控制闪烁——会导致时间不准
③进入设置状态后不关闭计时器中断——进入设置后虽然可以闪烁但是同时时钟秒数还是会增加

解决方案。。原来我蠢了,最后我突然想到了给秒数加1 加上一个条件,当进入设置状态后秒数不在加一,我也不用把定时器关闭。。

三、尚未实现的问题:

显示日期的月份天数的调整,一个月多少天的问题没有处理,在此默认一个月30天

四、仿真图:

在这里插入图片描述

五、源代码:

#include "reg52.h"
#include "intrins.h"

typedef unsigned int u16;	  //对数据类型进行声明定义
typedef unsigned char u8;

//定义P0口的三个引脚,赋予不同的涵义
sbit SER = P0^1;    //p0.1脚控制串行数据输入
sbit SCK = P0^0;    //串行输入时钟
sbit RCK = P0^2;    //存储寄存器时钟

sbit button1 = P2^0; 	//设置/确认按键
sbit button2 = P2^1;	//加按键
sbit button3 = P2^2;	//减按键
sbit button4 = P2^3;	//查看日期按键	暂时没有实现

//定义按键按下的次数,不同次数选择不同设置
static char button_num1 = 0; 	//判断选则时分秒
static char button_num2 = 0;	//判断切换时间/日期		//还没有处理日期设置和月天数判断


u8 code smgduan[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
					0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//显示0~F的值
char duanMa[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};//选择1-8哪个数码管段
char duanZhi[8]={0,0,0x40,0,0,0x40,0,0};	 	//保存8个数码管的中每个数码管段的数值 0x40:显示横杠

//num1:秒初始值 num2:分初始值 num3:时初始值
u16 num1=55,num2=59,num3=23;

u16 year=20,month=1,day=29; 
u8 day_flag = 0;	//一个月多少天的标志 flag=0 30一个月 flag=1 31一个月
u8 setting_flag = 0;	//进入设置位标志,进入后相应数字位闪烁 0:不闪烁	1:闪烁
//u8 day_time_flag = 0;	//显示时间/日期标志	0:显示时间	1:显示日期

static int i = 0;	//给中断计数使用

//函数声明
void SendTo595(char byteData);


/***********************************************************
*函数名		:delay_us
*功能		:延时
*参数		:i 延时的us数
************************************************************/
void delay_us(int i)
{
	while(i--);
}

/***********************************************************
*函数名		:display
*功能		:对传如的时分秒进行处理计算,转化为七段数码管要显示的值
*参数		:num1 秒	num2 分	  num3 时
************************************************************/
void display(u16 shi,u16 fen,u16 miao,u16 year,u16 month,u16 day)
{
	//先发送8位位码,后发送8位段码
	//8位数码管需要发送8次
	char i=0;	//给单片机for循环使用,由于Keil4 把 变量定义放for里会报错,只能放函数体前面
	static char shi1,ge1,shi2,ge2,shi3,ge3;

	//分离每个时间数字的个位和十位/
	if(button_num2 == 0)
	{
		shi1=(char)shi/10;
		ge1=(char)shi%10;
	
		shi2=(char)fen/10;
		ge2=(char)fen%10;
	
		shi3=(char)miao/10;
		ge3=(char)miao%10;	
	}
	//分离每个日期数字的个位和十位/
	else if(button_num2 == 1)
	{
		shi1=(char)year/10;
		ge1=(char)year%10;
	
		shi2=(char)month/10;
		ge2=(char)month%10;
	
		shi3=(char)day/10;
		ge3=(char)day%10;	
	}
	=======/

	//保存段值/
	duanZhi[0]=smgduan[shi1];
	duanZhi[1]=smgduan[ge1];
	duanZhi[3]=smgduan[shi2];
	duanZhi[4]=smgduan[ge2];
	duanZhi[6]=smgduan[shi3];
	duanZhi[7]=smgduan[ge3];	
	///=======

	//处理设置位的闪烁
	switch(button_num1)
	{
		case 0: break;
		case 1:	
		if(setting_flag ==1)  //被设置位灭0.5秒
		{
			duanZhi[0]=0;
			duanZhi[1]=0;
		} 
		break;	
		case 2:	
		if(setting_flag == 1) 
		{
			duanZhi[3]=0;
			duanZhi[4]=0;
		} 
		break;
		case 3: 
		if(setting_flag == 1)
		{
			duanZhi[6]=0;
			duanZhi[7]=0;
		} 		 
		break;
		default: break;
	}


	i=0;
 	for(;i<8;i++)	
	{
		SendTo595(~duanMa[i]); 		//送段码
		SendTo595(duanZhi[i]);		//送位码

		
		/*位移寄存器数据准备完毕,转移到存储寄存器*/
	   RCK = 0;         
	   _nop_();
	   _nop_();
	   RCK = 1; 	   //检测到上升沿,让存储寄存器时钟变为高电平
	}		
}


/*******************************************************************************
* 函 数 名         : TimerInit
* 函数功能		   : 定时器0初始化
* 参数			   :无
*******************************************************************************/
void TimerInit()
{
	TMOD|=0X01;	//选择为定时器0模式,工作方式1,仅用TR0打开启动。
	TH0=0X3C;	//给定时器赋初值,定时50ms 		3CB0
	TL0=0XB0;	//0X3CB0的十进制是15536 从15536计数到65536计数50000次 即50000us=50ms	
	ET0=1;		//打开定时器0中断允许
	EA=1;		//打开总中断
	TR0=1;		//打开定时器	
}


/*******************************************************************************
* 函 数 名         : button_setting
* 函数功能		   : 实现按键设置/确认功能,按四下设置完毕 重新开始计数
* 参数			   :无
*******************************************************************************/
void button_setting()
{	
	if(button1 == 0) 
	{
		delay_us(10);
		while(!button1);	//直到松开才向下执行

		button_num1++;	//选择设置不同位(时 分 秒)
		if(button_num1 == 4)
		{
			button_num1 = 0;
//			EA = 1;		//开中断 	//不需要关闭中断,只需要在秒数加一前添加一个条件即可防止在设置时秒数自增
		}
		else
		{
//			EA=0;	//关闭总中断,停止计时
		}	
	}
}

/*******************************************************************************
* 函 数 名         : button_data
* 函数功能		   : 时间/日期 切换显示按钮实现
* 参数			   :无
*******************************************************************************/
void button_data()
{
	if(button4 == 0)
	{
		delay_us(10);
		while(!button4);
		button_num2++;
		if(button_num2 == 2)	
		{
			button_num2=0;
		}
	}
}

/*******************************************************************************
* 函 数 名         : button_up_down
* 函数功能		   : 时间加/减 按键逻辑处理
* 参数			   :无
*******************************************************************************/
void button_up_down()
{
	if(button2 == 0)
	{
		switch(button_num1)
		{
			case 0:	break;
//			case 1:	delay_us(10); while(~button2); num3++;  break; 		//等待按键释放	时加一
//			case 2:	delay_us(10); while(~button2); num2++;	break;		//等待按键释放	分加一
//			case 3:	delay_us(10); while(~button2); num1++;	break;		//等待按键释放	秒加一
			case 1:	delay_us(10); while(~button2); 	   //等待按键释放	时加一
			if(button_num2 == 0)
			{
				num3++;
			} 
			else
			{
				year++;
			}
			 break; 		
			case 2:	delay_us(10); while(~button2); 	  	//等待按键释放	分加一
			if(button_num2 == 0)
			{
				num2++;
			} 
			else
			{
				month++;
			}
			 break; 	
			case 3:	delay_us(10); while(~button2);		 //等待按键释放	秒加一
			 if(button_num2 == 0)
			{
				num1++;
			} 
			else
			{
				day++;
			}
			 break;		
			default: break;	
		}
		if(num3 == 24) num3 = 0;	//时间和日期超出归零处理
		if(num2 == 60) num2 = 0;
		if(num1 == 60) num1 = 0;
		if(year == 100) year = 0;
		if(month == 13 ) month = 1;
		if(day == 31) day = 1;
		
	}

	if(button3 == 0)
	{
		switch(button_num1)
		{
			case 0:		break;
//			case 1:	delay_us(10);  num3--; while(~button3); break;	//时减一
//			case 2:	delay_us(10);  num2--; while(~button3);	break;	//分减一
//			case 3:	delay_us(10);  num1--; while(~button3);	break;	//秒减一
			case 1:	delay_us(10); while(~button3); 	   //等待按键释放	时加一
			if(button_num2 == 0)
			{
				num3--;
			} 
			else
			{
				year--;
			}
			 break; 		
			case 2:	delay_us(10); while(~button3); 	  	//等待按键释放	分加一
			if(button_num2 == 0)
			{
				num2--;
			} 
			else
			{
				month--;
			}
			 break; 	
			case 3:	delay_us(10); while(~button3);		 //等待按键释放	秒加一
			 if(button_num2 == 0)
			{
				num1--;
			} 
			else
			{
				day--;
			}
			 break;	
			default: break;
		}
		if(num3 == -1) num3 = 23;	//时间和日期减到低循环减
		if(num2 == -1) num2 = 59;
		if(num1 == -1) num1 = 59;
		if(year == -1) year = 99;
		if(month == 0 ) month = 12;
		if(day == 0) day = 30;
	}
}

/*******************************************************************************
* 函 数 名       : data_deal
* 函数功能		 : 日期处理函数,计算日期的当前的日期值
* 参数			 :无
*******************************************************************************/
void data_deal()
{

	if(day>=30)
	{
		if(day_flag==0)
		{
			month++;
			day=1;
		}
		if(day_flag==1&&day>=31)	
		{
			month++;	//此段不会执行,未处理
			day=1;
		}

		if(month>=13)
		{
			month = 0;
			year++;
			if(year>=100)
			{
				year = 0;
			}
		}
			
	}
}

/*******************************************************************************
* 函 数 名       : main
* 函数功能		 : 主函数
* 参数			 :无
*******************************************************************************/	
void main()
{	
	TimerInit();
	
	while(1)
	{		
		button_setting();
		button_data();
		button_up_down();
		display(num3,num2,num1,year,month,day); 
			
	}
}


/***********************************************************
*函数名		:SendTo595
*功能		:串行发送8个比特(一个字节)的数据给595,再并行输出
*参数		:byteData 
************************************************************/
void SendTo595(char byteData)
{
   char i=0;
    for(;i<8;i++)
   {
        SER = byteData>>7; 		//取出最高位(第8位)       
        byteData= byteData<<1;  //将第7位移动到最高位    

        SCK = 0;        //变为低电平,为下次准备  ,并延时2个时钟周期 
        _nop_();
        _nop_();

        SCK = 1;         //上升沿,让串行输入时钟变为高电平,
   }  
}


/*******************************************************************************
* 函 数 名         : Timer0()
* 函数功能		   : 定时器0中断函数
* 参数			   :无
*******************************************************************************/		
void Timer0() interrupt 1
{	 
	 TH0=0x3C;
	 TL0=0xB0;
	 i++;	 
	 //之前没有加入按键的时候我是在主函数中检测时间加一的,
	 //但是我后来发现如果我按住按键不松开的话时间计数会暂停
	 //其次我把所有判断都从if(==)替换为if(>=) 
	 //因为按键按住不松开,计数i的值会超过20,从而无法归零,导致时间计数停止,而i最终将会溢出

	 //说明:中断函数里是不适宜放过多语句的,这也是一开始我没有将时间加一处理放在中断的原因

	//使得setting_flag 0.2秒转换一次
	if(i%10<5)
	{
		setting_flag = 0;
	}
	if(i%10>=5)
	{
		setting_flag = 1;
	}

	 if(i>=20)//20个50毫秒即一秒
		{			
			i=0;
			if(button_num1==0)	//
			{
				num1++;	
			}
			
			if(num1>=60)
			{
				num1=0;
				num2++;
				if(num2>=60)//定时一小时自动清零
				{
					num2=0;
					num3++;
					if(num3>=24)
					{
						num3=0;
						day++;
						//日期处理
						data_deal();				
					}
				}	
			}
		}		
}

来个动图演示:
在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值