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 1001 | 9 |
0001 0000 | 16 |
导致进位不正确,所以我们需要将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位数据
- 这句代码通过“按位与”以及“按位左移”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
结果显示