51 单片机[10]:DS1302时钟

可调时钟完整代码已上传:https://github.com/TiezhuXing01/STC89C52

目标

  1. 在LCD1602上显示时钟,包括:年、月、日、时、分、秒
  2. 给时钟加上按键调时功能

1. DS1302

DS1302是由美国DALLAS公司推出的具有涓细电流充电能力的低功耗实时时钟芯片。它可以对年、月、日、周、时、分、秒进行计时,且具有闰年补偿等多种功能
RTC(Real Time Clock):实时时钟,是一种集成电路,通常称为时钟芯片

1.1 引脚定义和应用电路

img

img

1.2 内部结构框图

img

1.3 寄存器定义

img

1.4 时序定义

img

2. 时钟

新建一个项目。因为需要LCD1602显示时钟,所以要把LCD1602.c和LCD1602.h复制过来。新建DS1302.c和DS1302.h。

首先写DS1302.c文件。因为要对SCLK、I/O、CE三个端口进行操作,所以首先要对它们进行定义。

img

sbit DS1302_SCLK = P3^6;
sbit DS1302_IO = P3^4;
sbit DS1302_CE = P3^5;

2.1 初始化

根据时序定义,首先要定义一个初始化函数把CE和SCLK置0。然后定义一个单字节写和单字节读.

初始化函数:

void DS1302_Init()
{
	DS1302_CE = 0;
	DS1302_SCLK = 0;
}

2.2 单字节写

首先看第一个时序,CE要置1,IO口要传入第0个指令,SCLK要先置1再置0。
img

void DS1302_WriteByte(unsigned char Command, Data)
{
	DS1302_CE = 1;
	DS1302_IO = Command&0x01;
	DS1302_SCLK = 1;
	DS1302_SCLK = 0;
}

第二步要把第1位指令传给IO口,SCLK先置1再置0。
img

void DS1302_WriteByte(unsigned char Command, Data)
{
	DS1302_CE = 1;
	DS1302_IO = Command&0x01;
	DS1302_SCLK = 1;
	DS1302_SCLK = 0;
	
	DS1302_IO = Command&0x02;
	DS1302_SCLK = 1;
	DS1302_SCLK = 0;
}

这样一位一位写太麻烦了,可以用一个for循环实现。

void DS1302_WriteByte(unsigned char Command, Data)
{
	unsigned char i;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&0x01<<i;
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
	}
}

此时,时序图到中间了
img

前8位传的是指令,后8位将要传数据。可以用同样的方法(for循环),传数据。

void DS1302_WriteByte(unsigned char Command, Data)
{
	unsigned char i;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&0x01<<i;
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
	}
	for(i=0;i<8;i++)
	{
		DS1302_IO = Data&0x01<<i;
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
	}
}

此时,到了时序图的末尾
img

最后一步:需要将CE置0。

void DS1302_WriteByte(unsigned char Command, Data)
{
	unsigned char i;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&0x01<<i;
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
	}
	for(i=0;i<8;i++)
	{
		DS1302_IO = Data&0x01<<i;
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
	}
	DS1302_CE = 0;
}

2.3 单字节读

读操作的第一步:CE置1,SCLK置1,I/O口传入第0个指令。

img

第二步:SCLK先置0再置1,I/O口传入第1个指令。

img

前8位命令可以用一个类似之前写操作的for循环读取:

unsigned char DS1302_ReadByte(unsigned char Command)
{
	unsigned char i;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&0x01<<i;
		DS1302_SCLK = 0;
		DS1302_SCLK = 1;
	}
}

此时,到了时序图的中间

img

接着,可以用for循环读取D0~D7

img

unsigned char DS1302_ReadByte(unsigned char Command)
{
	unsigned char i, Data=0x00;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&(0x01<<i);
		DS1302_SCLK = 0;
		DS1302_SCLK = 1;
	}
	for(i=0;i<8;i++)
	{
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
		if(DS1302_IO){Data|=(0x01<<i);}
	}
	DS1302_CE = 0;
	return Data;
}

然后把刚才定义的三个函数在DS1302.h中声明

#ifndef __DS1302__H__
#define __DS1302__H__

void DS1302_Init();
void DS1302_WriteByte(unsigned char Command, Data);
unsigned char DS1302_ReadByte(unsigned char Command);

#endif

2.4 验证函数正确性

下面,在main.c中加入秒钟,验证一下刚刚写的函数有没有错误。

#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "Delay.h"

unsigned char Second;

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"RTC");
	
	DS1302_WriteByte(0x80,0x03);
	
	while(1)
	{
		Second = DS1302_ReadByte(0x81);
	  LCD_ShowNum(2, 1, Second, 3);
	}
}

编译一下,发现如图所示现象,数字保持在255不动。

img

这说明DS1302处于写保护,要加一行代码解除写保护。

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"RTC");
	DS1302_WriteByte(0x8E,0x00);//解除写保护
	DS1302_WriteByte(0x80,0x03);
	
	while(1)
	{
        Second = DS1302_ReadByte(0x81);
	    LCD_ShowNum(2, 1, Second, 3);
	}
}

同时,还要在单字节写函数中加一行代码

unsigned char DS1302_ReadByte(unsigned char Command)
{
    unsigned char i, Data=0x00;
	DS1302_CE = 1;
	for(i=0;i<8;i++)
    //此处省略部分代码...
    DS1302_CE = 0;
	DS1302_IO = 0;//新增本行代码
	return Data;
}

新增这行代码表示,在读操作完成后将I/O口清零。

编译一下,发现数字是会变化的,但不是一秒一秒变化,而且没有规律,有时数字显示会花。

这是因为DS1302里面的数是BCD码。

2.5 BCS码

BCD码(Binary Coded Decimal),用4位二进制数来表示1位十进制数

例:0001 0011表示13,1000 0101表示85,0001 1010不合法

在十六进制中的体现:0x13表示13,0x85表示85,0x1A不合法

BCD码转十进制:DEC=BCD/16*10+BCD%16; (2位BCD)

十进制转BCD码:BCD=DEC/10*16+DEC%10; (2位BCD)

2.6 转换为十进制显示

修改LCD_ShowNum里的参数即可。

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"RTC");
	DS1302_WriteByte(0x8E,0x00);//解除写保护
	DS1302_WriteByte(0x80,0x03);
	
	while(1)
	{
		Second = DS1302_ReadByte(0x81);
	    LCD_ShowNum(2, 1, Second/16*10+Second%16, 3);
	}
}

2.7 加入分钟

unsigned char Second, Minute;

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"RTC");
	DS1302_WriteByte(0x8E,0x00);//解除写保护
	DS1302_WriteByte(0x80,0x55);
	
	while(1)
	{
		Second = DS1302_ReadByte(0x81);
		Minute = DS1302_ReadByte(0x83);
		LCD_ShowNum(2, 4, Second/16*10+Second%16, 2);
		LCD_ShowString(2, 3, ":");
		LCD_ShowNum(2, 1, Minute/16*10+Minute%16, 2);
	}
}

编译一下,发现秒钟59之后,分钟进一。
img

2.8 优化代码,加入年月日星期

同时显示年、月、日、时、分、秒、星期,需要定义7个变量,很不方便,可以用一个数组简化变量定义。

在DS1302.c中定义一个数组

unsigned char DS1302_Time[] = {24, 7, 29, 10, 49, 55, 1};

然后还需要定义一个设置时间函数和一个读取时间函数。

void DS1302_SetTime(void)
{

}

void DS1302_ReadTime(void)
{
	
}

在设定时间之前,要定义一个 写入 地址表格
img

#define DS1302_SECOND    0x80
#define DS1302_MINUTE    0x82
#define DS1302_HOUR      0x84
#define DS1302_DATE      0x86
#define DS1302_MONTH     0x88
#define DS1302_DAY       0x8A
#define DS1302_YEAR      0x8C
#define DS1302_WP        0x8E//写保护

写入地址或0x01就是读的地址,无需再定义一遍读的地址。
0x80 | 0x01 = 1000 0000 | 0000 0001 = 1000 0001 = 0x81
0x82 | 0x01 = 1000 0010 | 0000 0001 = 1000 0011 = 0x83
0x84 | 0x01 = 1000 0100 | 0000 0001 = 1000 0101 = 0x85
以此类推……

这样就可以用同一个指令控制读写了。
所以单字节读函数就可以修改一下

unsigned char DS1302_ReadByte(unsigned char Command)
{
	unsigned char i, Data=0x00;
	Command |= 0x01;//写指令转换为读指令
	DS1302_CE = 1;
	for(i=0;i<8;i++)
	{
		DS1302_IO = Command&(0x01<<i);
		DS1302_SCLK = 0;
		DS1302_SCLK = 1;
	}
	for(i=0;i<8;i++)
	{
		DS1302_SCLK = 1;
		DS1302_SCLK = 0;
		if(DS1302_IO){Data|=(0x01<<i);}
	}
	DS1302_CE = 0;
	DS1302_IO = 0;
	return Data;
}
2.8.1 设置时间函数

下面开始定义设置时间函数。在设置时间时要把十进制数转为BCD码。

void DS1302_SetTime(void)
{
    DS1302_WriteByte(DS1302_WP,0x00);//关闭写保护
    DS1302_WriteByte(DS1302_YEAR,DS1302_Time[0]/10*16+DS1302_Time[0]%10);
    DS1302_WriteByte(DS1302_MONTH,DS1302_Time[1]/10*16+DS1302_Time[1]%10);
	DS1302_WriteByte(DS1302_DATE,DS1302_Time[2]/10*16+DS1302_Time[2]%10);
	DS1302_WriteByte(DS1302_HOUR,DS1302_Time[3]/10*16+DS1302_Time[3]%10);
	DS1302_WriteByte(DS1302_MINUTE,DS1302_Time[4]/10*16+DS1302_Time[4]%10);
	DS1302_WriteByte(DS1302_SECOND,DS1302_Time[5]/10*16+DS1302_Time[5]%10);
	DS1302_WriteByte(DS1302_DAY,DS1302_Time[6]/10*16+DS1302_Time[6]%10);
	DS1302_WriteByte(DS1302_WP,0x80);//打开写保护
}
2.8.2 读取时间函数

在读取时间时,要把BCD码转为十进制数。

void DS1302_ReadTime(void)
{
	unsigned char Temp;
	Temp=DS1302_ReadByte(DS1302_YEAR);
	DS1302_Time[0]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_MONTH);
	DS1302_Time[1]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_DATE);
	DS1302_Time[2]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_HOUR);
	DS1302_Time[3]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_MINUTE);
	DS1302_Time[4]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_SECOND);
	DS1302_Time[5]=Temp/16*10+Temp%16;
	Temp=DS1302_ReadByte(DS1302_DAY);
	DS1302_Time[6]=Temp/16*10+Temp%16;
}

然后在DS1302.h中声明一下这两个函数和设置时间的数组。

#ifndef __DS1302__H__
#define __DS1302__H__

extern unsigned char DS1302_Time[];

void DS1302_Init();
void DS1302_WriteByte(unsigned char Command, Data);
unsigned char DS1302_ReadByte(unsigned char Command);
void DS1302_SetTime(void);
void DS1302_ReadTime(void);

#endif

2.9 修改main.c

现在main.c就可以修改为:

#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "Delay.h"


void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");


	DS1302_SetTime();
	
	while(1)
	{
		DS1302_ReadTime();
		LCD_ShowNum(1,1,DS1302_Time[0],2);//年
		LCD_ShowNum(1,4,DS1302_Time[1],2);//月
		LCD_ShowNum(1,7,DS1302_Time[2],2);//日
		LCD_ShowNum(2,1,DS1302_Time[3],2);//时
		LCD_ShowNum(2,4,DS1302_Time[4],2);//分
		LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	}
}

编译一下,就可以看到如图所示现象
img

3. 按键可调时钟

新建项目。我们在按键控制LED流水灯项目中写过Key.c和Key.h,把它俩复制过来。闪烁功能需要用到定时器,所以要把之前定时器时钟项目写的Timer0.c和Timer0.h复制过来。

先测试一下Key()好不好使

#include "Delay.h"
#include "Key.h"
#include "Timer0.h"

unsigned char KeyNum;

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");

  
	DS1302_SetTime();
	
	while(1)
	{
		KeyNum=Key();
		if(KeyNum)
		{
			LCD_ShowNum(2,10,KeyNum,2);
		}
		DS1302_ReadTime();
		LCD_ShowNum(1,1,DS1302_Time[0],2);//年
		LCD_ShowNum(1,4,DS1302_Time[1],2);//月
		LCD_ShowNum(1,7,DS1302_Time[2],2);//日
		LCD_ShowNum(2,1,DS1302_Time[3],2);//时
		LCD_ShowNum(2,4,DS1302_Time[4],2);//分
		LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	}
}

编译一下,发现按下K1显示01,按下K2显示02,按下K3显示03,按下K4显示04。

要根据按键切换MODE,根据MODE改变函数。

修改main.c

#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "Delay.h"
#include "Key.h"
#include "Timer0.h"

unsigned char KeyNum,MODE;

void TimeShow()
{
		DS1302_ReadTime();
		LCD_ShowNum(1,1,DS1302_Time[0],2);//年
		LCD_ShowNum(1,4,DS1302_Time[1],2);//月
		LCD_ShowNum(1,7,DS1302_Time[2],2);//日
		LCD_ShowNum(2,1,DS1302_Time[3],2);//时
		LCD_ShowNum(2,4,DS1302_Time[4],2);//分
		LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
}

void TimeSet()
{
	P2=0x00;//只做测试用
}

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");

  
	DS1302_SetTime();
	
	while(1)
	{
		KeyNum=Key();
		if(KeyNum==1)
		{
			if(MODE==0){MODE=1;}
			else if(MODE==1){MODE=0;}
		}
		switch(MODE)
		{
			case 0:TimeShow();break;
			case 1:TimeSet();break;
		}
	}
}

编译一下,时钟正常运行(程序在TimeShow()),按下K1,时钟暂停LED全亮(程序在TimeSet())。

下面编写TimeSet()函数。

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		if(TimeSetSelect>6){TimeSetSelect=0;}
	}
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

其中if(TimeSetSelect>6){TimeSetSelect=0;}可以简化为TimeSetSelect%=6;
TimeSetSelect++;TimeSetSelect%=6;可以合并为TimeSetSelect++%=6;。但为了方便理解,可以不这样合并。

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect%=6;
	}
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

编译一下,发现按下K1时钟暂停,LCD第2行第10列显示00,按下K2后00变01,最高变到05再按K2就清零。

增加选择年月日时分秒增减功能

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect%=6;
	}
	if(KeyNum==3)
	{
		DS1302_Time[TimeSetSelect]++;
	}
	if(KeyNum==4)
	{
		DS1302_Time[TimeSetSelect]--;
	}
	//更新显示
	LCD_ShowNum(1,1,DS1302_Time[0],2);//年
	LCD_ShowNum(1,4,DS1302_Time[1],2);//月
	LCD_ShowNum(1,7,DS1302_Time[2],2);//日
	LCD_ShowNum(2,1,DS1302_Time[3],2);//时
	LCD_ShowNum(2,4,DS1302_Time[4],2);//分
	LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

编译一下。发现按下K1时钟暂停,当第2行第10列显示00时,按K3和K4可以增减年;当第2行第10列显示01时,按K3和K4可以增减月;以此类推。但是再按K1就又返回初始时间了,没有更新时间。

3.1 过大越界检查

下面加入越界检查代码。首先加入增越界判断:

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect%=6;
	}
	if(KeyNum==3)
	{
		DS1302_Time[TimeSetSelect]++;
		if(DS1302_Time[0]>99){DS1302_Time[0]=0;}//年越界判断
		if(DS1302_Time[1]>12){DS1302_Time[1]=1;}//月越界判断
		//大月31天
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 ||
			DS1302_Time[1]==7 || DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
		}
		//小月30天
		else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 ||
			DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
		}
		//2月29或28天
		else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%4==0)//平闰年判断
			{
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
			}
			else
			{
				if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
			}
		}
		if(DS1302_Time[3]>23){DS1302_Time[3]=0;}//时越界判断
		if(DS1302_Time[4]>59){DS1302_Time[4]=0;}//分越界判断
		if(DS1302_Time[5]>59){DS1302_Time[5]=0;}//秒越界判断
	}
	if(KeyNum==4)
	{
		DS1302_Time[TimeSetSelect]--;
	}
	//更新显示
	LCD_ShowNum(1,1,DS1302_Time[0],2);//年
	LCD_ShowNum(1,4,DS1302_Time[1],2);//月
	LCD_ShowNum(1,7,DS1302_Time[2],2);//日
	LCD_ShowNum(2,1,DS1302_Time[3],2);//时
	LCD_ShowNum(2,4,DS1302_Time[4],2);//分
	LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

编译一下,可以看到年到99之后再加就变00了,月最高到12月再加就变01了,日会跟随大小月、平闰年变化。时分秒也不会越界。
但是减到00后再减会变成55,这其实是变成了255,但是只显示了2位。所以就需要加入减越界判断。

3.2 过小越界检查

下面加入减越界判断。由于之前定义的时间数组是无符号的,不方便越界判断。为了方便越界判断,要把unsigned char DS1302_Time[]改为char DS1302_Time[]

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect%=6;
	}
	if(KeyNum==3)
	{
		DS1302_Time[TimeSetSelect]++;
		if(DS1302_Time[0]>99){DS1302_Time[0]=0;}//年越界判断
		if(DS1302_Time[1]>12){DS1302_Time[1]=1;}//月越界判断
		//大月31天
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 ||
			DS1302_Time[1]==7 || DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
		}
		//小月30天
		else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 ||
			DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
		}
		//2月29或28天
		else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%4==0)//平闰年判断
			{
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
			}
			else
			{
				if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
			}
		}
		if(DS1302_Time[3]>23){DS1302_Time[3]=0;}//时越界判断
		if(DS1302_Time[4]>59){DS1302_Time[4]=0;}//分越界判断
		if(DS1302_Time[5]>59){DS1302_Time[5]=0;}//秒越界判断
	}
	if(KeyNum==4)
	{
		DS1302_Time[TimeSetSelect]--;
		if(DS1302_Time[0]<0){DS1302_Time[0]=99;}//年越界判断
		if(DS1302_Time[1]<1){DS1302_Time[1]=12;}//月越界判断
		//大月31天
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 ||
			DS1302_Time[1]==7 || DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=31;}
		}
		//小月30天
		else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 ||
			DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=30;}
		}
		//2月29或28天
		else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%4==0)//平闰年判断
			{
				if(DS1302_Time[2]<1){DS1302_Time[2]=29;}
			}
			else
			{
				if(DS1302_Time[2]<1){DS1302_Time[2]=28;}
			}
		}
		if(DS1302_Time[3]<1){DS1302_Time[3]=23;}//时越界判断
		if(DS1302_Time[4]<0){DS1302_Time[4]=59;}//分越界判断
		if(DS1302_Time[5]<0){DS1302_Time[5]=59;}//秒越界判断
	}
	//更新显示
	LCD_ShowNum(1,1,DS1302_Time[0],2);//年
	LCD_ShowNum(1,4,DS1302_Time[1],2);//月
	LCD_ShowNum(1,7,DS1302_Time[2],2);//日
	LCD_ShowNum(2,1,DS1302_Time[3],2);//时
	LCD_ShowNum(2,4,DS1302_Time[4],2);//分
	LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

编译一下,可以看到年月日时分秒全都在合法范围内增减。

但是还有一个问题,当月被调成12月,日被调成31后,再把月调到11月,日没有变化,还是31日。这是因为我们没有在日过小越界中判断月份不同情况下的日过大越界。下面做一下调整。

void TimeSet()
{
	if(KeyNum==2)
	{
		TimeSetSelect++;
		TimeSetSelect%=6;
	}
	if(KeyNum==3)
	{
		DS1302_Time[TimeSetSelect]++;
		if(DS1302_Time[0]>99){DS1302_Time[0]=0;}//年越界判断
		if(DS1302_Time[1]>12){DS1302_Time[1]=1;}//月越界判断
		//大月31天
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 ||
			DS1302_Time[1]==7 || DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
		}
		//小月30天
		else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 ||
			DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
		}
		//2月29或28天
		else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%4==0)//平闰年判断
			{
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
			}
			else
			{
				if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
			}
		}
		if(DS1302_Time[3]>23){DS1302_Time[3]=0;}//时越界判断
		if(DS1302_Time[4]>59){DS1302_Time[4]=0;}//分越界判断
		if(DS1302_Time[5]>59){DS1302_Time[5]=0;}//秒越界判断
	}
	if(KeyNum==4)
	{
		DS1302_Time[TimeSetSelect]--;
		if(DS1302_Time[0]<0){DS1302_Time[0]=99;}//年越界判断
		if(DS1302_Time[1]<1){DS1302_Time[1]=12;}//月越界判断
		//大月31天
		if(DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 ||
			DS1302_Time[1]==7 || DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=31;}
			if(DS1302_Time[2]>31){DS1302_Time[2]=1;}//新增过大越界判断
		}
		//小月30天
		else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 ||
			DS1302_Time[1]==11)
		{
			if(DS1302_Time[2]<1){DS1302_Time[2]=30;}
			if(DS1302_Time[2]>30){DS1302_Time[2]=1;}//新增过大越界判断
		}
		//2月29或28天
		else if(DS1302_Time[1]==2)
		{
			if(DS1302_Time[0]%4==0)//平闰年判断
			{
				if(DS1302_Time[2]<1){DS1302_Time[2]=29;}
				if(DS1302_Time[2]>29){DS1302_Time[2]=1;}//新增过大越界判断
			}
			else
			{
				if(DS1302_Time[2]<1){DS1302_Time[2]=28;}
				if(DS1302_Time[2]>28){DS1302_Time[2]=1;}//新增过大越界判断
			}
		}
		if(DS1302_Time[3]<1){DS1302_Time[3]=23;}//时越界判断
		if(DS1302_Time[4]<0){DS1302_Time[4]=59;}//分越界判断
		if(DS1302_Time[5]<0){DS1302_Time[5]=59;}//秒越界判断
	}
	//更新显示
	LCD_ShowNum(1,1,DS1302_Time[0],2);//年
	LCD_ShowNum(1,4,DS1302_Time[1],2);//月
	LCD_ShowNum(1,7,DS1302_Time[2],2);//日
	LCD_ShowNum(2,1,DS1302_Time[3],2);//时
	LCD_ShowNum(2,4,DS1302_Time[4],2);//分
	LCD_ShowNum(2,7,DS1302_Time[5],2);//秒
	LCD_ShowNum(2,10,TimeSetSelect,2);
}

编译一下。在12月31日时,调整月份为11月后,日会自动从31变为01。

现在调好时间后,按下K1又变回原来的时间,这是因为没有写入更改后的时间。需要修改一下主函数

void main()
{
	LCD_Init();
	DS1302_Init();
	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");

  
	DS1302_SetTime();
	
	while(1)
	{
		KeyNum=Key();
		if(KeyNum==1)
		{
			if(MODE==0){MODE=1;}
			else if(MODE==1){MODE=0;DS1302_SetTime();}//新增了写入时间
		}
		switch(MODE)
		{
			case 0:TimeShow();break;
			case 1:TimeSet();break;
		}
	}
}

编译一下,现在设置好时间后按,K1退出设置模式,时间变为刚刚调整的时间。

3.3 选择闪烁功能

这一功能需要用到定时器。定义一个变量TimeSetFlashFlag

main()中加一行Timer0Init();

main.c下面加一个定时器中断函数

void Timer0_Rountine() interrupt 1
{
	static unsigned int T0Count;	//静态变量,退出函数后不丢失值
	T0Count++;
	TL0 = 0x18;		//重新赋初值,防止溢出后从0开始计数
	TH0 = 0xFC;
	if(T0Count>=500)	//每隔0.5秒执行一次
	{
		T0Count = 0;
		TimeSetFlashFlag=!TimeSetFlashFlag;
	}
}

测试一下。在void TimeSet()的“更新显示”代码段中加入

	LCD_ShowNum(2,13,TimeSetFlashFlag,2);

编译一下,发现按下K1进入设置模式后,LCD第2行第13列00和01交替变换。

下面加入闪烁代码。把void TimeSet()的“更新显示”代码段改为下面这样:

	//year
	if(TimeSetSelect==0 && TimeSetFlashFlag==1){LCD_ShowString(1,1,"  ");}
	else{LCD_ShowNum(1,1,DS1302_Time[0],2);}//年
	//month
	if(TimeSetSelect==1 && TimeSetFlashFlag==1){LCD_ShowString(1,4,"  ");}
	else{LCD_ShowNum(1,4,DS1302_Time[1],2);}//月
	//date
	if(TimeSetSelect==2 && TimeSetFlashFlag==1){LCD_ShowString(1,7,"  ");}
	else{LCD_ShowNum(1,7,DS1302_Time[2],2);}//日
	//hour
	if(TimeSetSelect==3 && TimeSetFlashFlag==1){LCD_ShowString(2,1,"  ");}
	else{LCD_ShowNum(2,1,DS1302_Time[3],2);}//时
	//minute
	if(TimeSetSelect==4 && TimeSetFlashFlag==1){LCD_ShowString(2,4,"  ");}
	else{LCD_ShowNum(2,4,DS1302_Time[4],2);}//分
	//second
	if(TimeSetSelect==5 && TimeSetFlashFlag==1){LCD_ShowString(2,7,"  ");}
	else{LCD_ShowNum(2,7,DS1302_Time[5],2);}//秒

编译一下,按下K1就可实现选择闪烁功能。

这个程序有一个小小的问题,就是按下K3或K4不松手,时间不会更新显示,这是因为程序此时卡在Key()的while循环中出不来了。

unsigned char Key()
{
	unsigned char KeyNumber=0;
	
	if(P3_1==0){Delay(20);while(P3_1==0);Delay(20);KeyNumber=1;}
	if(P3_0==0){Delay(20);while(P3_0==0);Delay(20);KeyNumber=2;}
	if(P3_2==0){Delay(20);while(P3_2==0);Delay(20);KeyNumber=3;}
	if(P3_3==0){Delay(20);while(P3_3==0);Delay(20);KeyNumber=4;}
	
	return KeyNumber;
}

还有一个小细节,如果不是第一次进入设置模式,那么进入设置模式后,闪烁的是上一次调整的位置,而不是默认“年”。

需要修改一下main()

void main()
{
	LCD_Init();
	DS1302_Init();
	Timer0Init();
	LCD_ShowString(1,1,"  -  -  ");
	LCD_ShowString(2,1,"  :  :  ");

  
	DS1302_SetTime();
	
	while(1)
	{
		KeyNum=Key();
		if(KeyNum==1)
		{
			if(MODE==0){MODE=1;TimeSetSelect=0;}//加上默认先改“年”
			else if(MODE==1){MODE=0;DS1302_SetTime();}
		}
		switch(MODE)
		{
			case 0:TimeShow();break;
			case 1:TimeSet();break;
		}
	}
}

以上就是关于DS1302时钟的全部内容。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值