51单片机【DS1302实时时钟模块化编程】(江协科技学习笔记)

DS1302是由美国DALLAS公司推出的具有涓细电流充电能力的低功耗实时时钟芯片。它可以对年、月、日、周、时、分、秒进行计时,且具有闰年补偿等多种功能。

RTC(Real Time Clock):实时时钟,是一种集成电路,通常称为时钟芯片。

单片机定时器时钟相较于DS1302的缺点:

1.精度不高

2.占用单片机CPU时间

3.不能掉电,否则自动清零,而DS1302具有备用电源。

32.768KHZ的晶振(标准的石英晶体振荡器):为时钟提供合适的精度,同时为实时时钟系统提供稳定的计数脉冲。

其频率正好是2的15次方,即32768赫兹。这个频率比较特殊,它是16位计数器的时钟周期,因此在电路设计中常常作为分频器的分频参数来使用。同时,这个频率也是全球通用的国际标准频率,它被广泛应用在各种计时器、电子钟表以及无线通信等领域中。

在实际的电路应用中,32.768khz晶振通常需要接上一个电容来稳定其振荡频率。因为石英晶体的振荡频率受到外界环境温度、机械振动等因素的影响,容易发生波动和漂移。而通过加入一个适当的电容,可以形成一个谐振回路,使得晶振的振荡频率在一定范围内保持稳定。

使能是芯片的一个输入引脚,或者电路的一个输入端口,只有该引脚激活,例如置于高电平时,整个模块才能正常工作。

使能端原理
当低电平有效时

使能信号为低电平时,电路实现与非门功能;

为高电平时,电路呈高阻态

当高电平有效时

使能信号为高电平时,电路实现与非功能;

为低电平时,电路实现高阻态

DS1302电路原理图

其中CE(使能)为高电平时,I/O,SCLK的操作无效;I/O控制移位寄存器。

地址位或控制位

第七位固定为1。

第六位给1,操作RAM;第六位给0,操作CK(clock时钟)。

第五位到第一位是地址(Address)。

第0位是读或写,RD(read),WR(write)上面加一横杠,意思是给0有效或低电平有效。

BIT0~BIT3是秒的个位,BIT4~BIT7是秒的十位,此处用BCD码的四位二进制表示一位的十进制。但

二进制十进制
0000 10019
0001 000016

导致进位不正确,所以我们需要将BCD码转化为十进制

时序定义

CE使能,在单字节的读/写过程中,CE始终保持高电平(给1),操作结束后清零。

SCLK,上升沿写入数据,下降沿输出数据。

I/O口,共两个字节,从左(即从最低位开始)到右写入,完成命令字。

单字符读(SINGLE-BYTE READ)有15个周期;

单字符写(SINGLE-BYTE WRITE)则有16个周期。

所以在编写时序时,需将单字符读(READ)补上一个周期。

编程操作:

1.给CE(使能)1

2.把数据放在IO口(数据输入输出)

3.给一个上升沿和下降沿

BCD

定义:用4位二进制数来表示1位十进制中的0~9这10个数码,是一种二进制的数字编码形式,用二进制编码的十进制代码。

补充:C语言中的运算符

C语言位运算符:与、或、异或、取反、左移和右移。

这些运算符只能用于整型操作数,即只能用于带符号或无符号的char,short,int与long类型。

& 按位与: 如果两个相应的二进制位都为1,则该位的结果值为1,否则为0

| 按位或: 两个相应的二进制位中只要有一个为1,该位的结果值为1

^ 按位异或: 若参加运算的两个二进制位值相同则为0,否则为1

~ 按位取反: ~是一元运算符,用来对一个二进制数按位取反,即将0变1,将1变0

<< 按位左移: 用来将一个数的各二进制位全部左移N位,右补0

>> 按位右移: 将一个数的各二进制位右移N位,移到右端的低位被舍弃,对于无符号数,高位补0!

DS1302_IO=Command&(0x01<<i);//解析

//借用按位与取出Command(八位数据)的第0位数据

  1. 这句代码通过“按位与”以及“按位左移”for循环的每一次循环,实现按序取出数据,如下

DS1302_IO=Command&0x01;//0000 0001

DS1302_SCLK=1;

DS1302_SCLK=0

DS1302_IO=Command&0x02;//0000 0010

DS1302_SCLK=1;

DS1302_SCLK=0;

DS1302_IO=Command&0x04;//0000 0100

DS1302_SCLK=1;

DS1302_SCLK=0;

数字全为65

数字乱码

解决办法,大家可以关注代码注释,希望能帮助到大家

main.c

//main.c
#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.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);//秒
		
	}
}

DS1302.c

//DS1302.C
#include <REGX52.H>

//重新对端口进行定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;

//寄存器写入地址/指令地址
#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

//时间数组
unsigned char DS1302_Time[]={23,11,21,14,10,32,1};//年,月,日,时,分,秒,星期

/**
  * @brief  DS1302初始化
  * @param  无
  * @retval 无
  */
void DS1302_Init(void)
{
	DS1302_CE=0;
	DS1302_SCLK=0;
}

/**
  * @brief  DS1302写一个字节
  * @param  Command 命令字/地址
  * @param  Data 要写入的数据
  * @retval 无
  */
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;//SCLK置1后立即置0,该时序操作需考虑时钟芯片是否可承受这个时钟的最快频率
		DS1302_SCLK=0;//由于单片机没有这么快的频率,故可不加延时
	}
	
	for(i=0;i<8;i++)
	{
		DS1302_IO=Data&(0x01<<i);
		DS1302_SCLK=1;
		DS1302_SCLK=0;
	}
	
	DS1302_CE=0;
}

/**
  * @brief  DS1302读一个字节
  * @param  Command 命令字/地址
  * @retval 读出的数据
  */
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;//先0后1,则全都是65

		if(DS1302_IO){Data|=(0x01<<i);}//“|或“,用于置1
	}
	DS1302_CE=0;
	DS1302_IO=0;//此处不将IO口置0,数字会出现乱码
	return Data;
}

/**
  * @brief  DS1302设置时间,调用之后,DS1302_Time数组的数字会被设置到DS1302中
  * @param  无
  * @retval 无
  */
void DS1302_SetTime(void)
{
	DS1302_WriteByte(DS1302_WP,0x00);//设置前把写保护关闭
	DS1302_WriteByte(DS1302_YEAR,	DS1302_Time[0]/10*16+DS1302_Time[0]%10);//年,十进制转换为BCD码
	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);
}

/**
  * @brief  DS1302读取时间,调用之后,DS1302中的数据会被读取到DS1302_Time数组中
  * @param  无
  * @retval 无
  */
void DS1302_ReadTime(void)
{
	unsigned char Temp;
	//读取的是BCD,需转为十进制
	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

//DS1302.h
#ifndef _DS1302_H_
#define _DS1302_H_

extern unsigned char DS1302_Time[];//年,月,日,时,分,秒
//extern的使用:	当变量声明外部变量,则前面必须加extern;函数和数组声明变量其默认有extern,可不用再添加
	
void DS1302_Init(void);
void DS1302_WriteByte(unsigned char Command,Data);
unsigned char DS1302_ReadByte(unsigned char Command);

void DS1302_SetTime(void);
void DS1302_ReadTime(void);

#endif

LCD1602.c

//LCD1602.c
#include <REGX52.H>

//引脚配置:
sbit LCD_RS=P2^6;
sbit LCD_RW=P2^5;
sbit LCD_EN=P2^7;
#define LCD_DataPort P0

//函数定义:
/**
  * @brief  LCD1602延时函数,12MHz调用可延时1ms
  * @param  无
  * @retval 无
  */
void LCD_Delay()
{
	unsigned char i, j;

	i = 2;
	j = 239;
	do
	{
		while (--j);
	} while (--i);
}

/**
  * @brief  LCD1602写命令
  * @param  Command 要写入的命令
  * @retval 无
  */
void LCD_WriteCommand(unsigned char Command)
{
	LCD_RS=0;
	LCD_RW=0;
	LCD_DataPort=Command;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602写数据
  * @param  Data 要写入的数据
  * @retval 无
  */
void LCD_WriteData(unsigned char Data)
{
	LCD_RS=1;
	LCD_RW=0;
	LCD_DataPort=Data;
	LCD_EN=1;
	LCD_Delay();
	LCD_EN=0;
	LCD_Delay();
}

/**
  * @brief  LCD1602设置光标位置
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @retval 无
  */
void LCD_SetCursor(unsigned char Line,unsigned char Column)
{
	if(Line==1)
	{
		LCD_WriteCommand(0x80|(Column-1));
	}
	else if(Line==2)
	{
		LCD_WriteCommand(0x80|(Column-1+0x40));
	}
}

/**
  * @brief  LCD1602初始化函数
  * @param  无
  * @retval 无
  */
void LCD_Init()
{
	LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵
	LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关
	LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动
	LCD_WriteCommand(0x01);//光标复位,清屏
}

/**
  * @brief  在LCD1602指定位置上显示一个字符
  * @param  Line 行位置,范围:1~2
  * @param  Column 列位置,范围:1~16
  * @param  Char 要显示的字符
  * @retval 无
  */
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char)
{
	LCD_SetCursor(Line,Column);
	LCD_WriteData(Char);
}

/**
  * @brief  在LCD1602指定位置开始显示所给字符串
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  String 要显示的字符串
  * @retval 无
  */
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=0;String[i]!='\0';i++)
	{
		LCD_WriteData(String[i]);
	}
}

/**
  * @brief  返回值=X的Y次方
  */
int LCD_Pow(int X,int Y)
{
	unsigned char i;
	int Result=1;
	for(i=0;i<Y;i++)
	{
		Result*=X;
	}
	return Result;
}

/**
  * @brief  在LCD1602指定位置开始显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~65535
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以有符号十进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:-32768~32767
  * @param  Length 要显示数字的长度,范围:1~5
  * @retval 无
  */
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length)
{
	unsigned char i;
	unsigned int Number1;
	LCD_SetCursor(Line,Column);
	if(Number>=0)
	{
		LCD_WriteData('+');
		Number1=Number;
	}
	else
	{
		LCD_WriteData('-');
		Number1=-Number;
	}
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0');
	}
}

/**
  * @brief  在LCD1602指定位置开始以十六进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~0xFFFF
  * @param  Length 要显示数字的长度,范围:1~4
  * @retval 无
  */
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i,SingleNumber;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		SingleNumber=Number/LCD_Pow(16,i-1)%16;
		if(SingleNumber<10)
		{
			LCD_WriteData(SingleNumber+'0');
		}
		else
		{
			LCD_WriteData(SingleNumber-10+'A');
		}
	}
}

/**
  * @brief  在LCD1602指定位置开始以二进制显示所给数字
  * @param  Line 起始行位置,范围:1~2
  * @param  Column 起始列位置,范围:1~16
  * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111
  * @param  Length 要显示数字的长度,范围:1~16
  * @retval 无
  */
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
	unsigned char i;
	LCD_SetCursor(Line,Column);
	for(i=Length;i>0;i--)
	{
		LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0');
	}
}

LCD1602.h

//LCD1602.h
#ifndef __LCD1602_H__
#define __LCD1602_H__

//用户调用函数:
void LCD_Init();
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char);
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String);
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void C(unsigned char Line,unsigned char Column,int Number,unsigned char Length);
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);

#endif

Delay.c(晶振为11.0592)

//Delay.c
void Delay(unsigned int xms)	
{
	unsigned char i, j;
	while(xms)
	{
		i=2;
		j=239;
		do
		{
			while(--j);
			}while(--i);
		xms--;
	}
}

Delay.h

//Delay.h
#ifndef _DELAY_H_
#define _DELAY_H_

void Delay(unsigned int xms);

#endif

结果显示

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例 基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例 基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例 基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例 基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例基于51单片机开发的小程序,学习资料,用于学习51单片机的嵌入式开发,应用案例

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值