当前位置: 首页 > news >正文

51单片机09 DS1302时钟

 

测试一 

测试代码:别忘了之前调整点阵的跳线

#include <STC89C5xRC.H>
#include "LCD1602.h"

void main()
{
	LCD_Init();
	LCD_ShowString(1,1,"RTC");
	while(1)
	{
		
	}
}

------------------------------------

测试二

DS1302.C

#include <STC89C5xRC.H>
#include "DS1302.h"

//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;


//如果给定的 SCLK 引脚在电路板上确实连接到 P3.6,那么可以直接操作 SCLK,代码会按照定义进行相应操作。
//这样定义是为了方便不同型号,只需要修改P36就可以。

void DS1302_Init(void)
{
	DS1302_CE=0; //手册中标注平时默认为0,CE只有在1的时候才工作
	DS1302_SCLK=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;
	}
	/*
	DS1302_IO=Command&0x01; //取出第0位
	DS1302_SCLK=1;
	DS1302_SCLK=0; 这里置1后立马置0需要考虑时钟芯片的操作时间(可能反应不过来)
	*/
	for(i=0;i<8;i++)
	{
		DS1302_IO=Data&(0x01<<i);
		DS1302_SCLK=1;
		DS1302_SCLK=0;
	}
	DS1302_CE=0;
	}

unsigned char DS1302_ReadByte(unsigned char Command)
{
	unsigned char i,Data=0x00;//局部变量初始化不一定是0,全局默认是
	DS1302_CE=1;
	for(i=0;i<8;i++)
	{
		DS1302_IO=Command&(0x01<<i);
		DS1302_SCLK=0;
		DS1302_SCLK=1; //一个脉冲控制两个,一个结束另一个直接开始,要调整顺序
		//第二部分的脉冲操作是芯片自己负责,自己不用操作,读出只需要置1
	}
	for(i=0;i<8;i++)
	{
		DS1302_SCLK=1; //最开始连续两次1,电平不会有变化(没有变化)
		DS1302_SCLK=0; 
		//如果是先0后1,搞完后SCLK是1,但是最后SCLK是0
		if(DS1302_IO){Data|=(0x01<<i);}
	}
	DS1302_CE=0;
	DS1302_IO=0;	//读取后将IO设置为0,否则读出的数据会出错
	return Data;
}

main.c

#include <STC89C5xRC.H>
#include "LCD1602.h"
#include "DS1302.h"

unsigned char second;

void main()
{
	LCD_Init();
	LCD_ShowString(1,1,"RTC");

	DS1302_WriteByte(0x8E,0x00);//前面一直显示128/129,加上这句好了
	DS1302_WriteByte(0x80,0x03);
	second=DS1302_ReadByte(0x81);
	
	LCD_ShowNum(2,1,second,3);
	while(1)
	{
		
	}
}

DS1302_WriteByte(0x8E,0x00):DS1302 实时时钟模块内部有一个控制寄存器0x8E,用于设置是否允许对数据进行读写。

DS1302 的秒数是以 BCD(Binary-Coded Decimal)格式存储的。

一、时钟

DS1302.C

#include <STC89C5xRC.H>

//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;
//如果给定的 SCLK 引脚在电路板上确实连接到 P3.6,那么可以直接操作 SCLK,代码会按照定义进行相应操作。
//这样定义是为了方便不同型号,只需要修改P36就可以。

//时间数组,索引0~6分别为年、月、日、时、分、秒、星期
unsigned char DS1302_Time[]={25,2,14,12,59,55,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


/**
  * @brief  DS1302初始化
  * @param  无
  * @retval 无
  */
void DS1302_Init(void)
{
	DS1302_CE=0; //手册中标注平时默认为0,CE只有在1的时候才工作
	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;
		DS1302_SCLK=0;
	}
	/*
	DS1302_IO=Command&0x01; //取出第0位
	DS1302_SCLK=1;
	DS1302_SCLK=0; 这里置1后立马置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;//局部变量初始化不一定是0,全局默认是
	Command|=0x01;//这里较测试2代码改了,写的地址+1就是读的地址,不用重复定义很多地址
	DS1302_CE=1;
	for(i=0;i<8;i++)
	{
		DS1302_IO=Command&(0x01<<i);
		DS1302_SCLK=0;
		DS1302_SCLK=1; //一个脉冲控制两个,一个结束另一个直接开始,要调整顺序
		//第二部分的脉冲操作是芯片自己负责,自己不用操作,读出只需要置1
	}
	for(i=0;i<8;i++)
	{
		DS1302_SCLK=1; //最开始连续两次1,电平不会有变化(没有变化)
		DS1302_SCLK=0; 
		//如果是先0后1,搞完后SCLK是1,但是最后SCLK是0
		if(DS1302_IO){Data|=(0x01<<i);}
	}
	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;
	Temp=DS1302_ReadByte(DS1302_YEAR);
	DS1302_Time[0]=Temp/16*10+Temp%16;//BCD码转十进制后读取
	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;
}

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 实时时钟芯片在初始化并设置好时间之后,会内部开始计时。每秒钟,芯片的 Second 寄存器会自动递增,从而导致在每次调用 DS1302_ReadTime() 时获取的值是自动更新的。

相关文章:

  • 人工智能之数学基础:线性空间
  • DeepSeek官方发布R1模型推荐设置
  • 重生之我在异世界学编程之C语言:深入指针篇(上)
  • 大数据SQL调优专题——Hive执行原理
  • Nginx--日志(介绍、配置、日志轮转)
  • Qt QOpenGLShaderProgram详解
  • 完美解决 error:0308010C:digital envelope routines::unsupported
  • ArcGIS Pro显示缓存空间不足导致编辑或加载数据显示不完全
  • duckdb导出Excel和导出CSV速度测试
  • 微信小程序 - 分包加载
  • 黑魔法(BMD)存储卡格式化后的恢复方法
  • 【IEEE/EI/CPCI检索】2025年第四届信号处理、信息系统与网络安全国际会议(SPISCS 2025)
  • vLLM启用笔记
  • 常用排序算法
  • SPSS—回归分析
  • 基于SpringBoot+Vue的科研项目验收管理系统的设计与实现(源码+SQL脚本+LW+部署讲解等)
  • React 第二十五节 <Fragment></Fragment> 的用途以及使用注意事项详解
  • 基于 MATLAB 的粒子滤波算法实现示例,用于处理手机传感器数据并估计电梯运行参数。
  • 回归预测 | Matlab实现PSO-HKELM粒子群算法优化混合核极限学习机多变量回归预测
  • MySQL篇之对MySQL进行参数优化,提高MySQL性能
  • 像淘宝购物网站建设需要哪些专业人员/百度seo搜索引擎优化培训
  • 互联网公司网站建设/长沙seo霜天博客
  • 高端建站骗局/seo排名优化代理
  • 荔湾区做网站公司/广州外包网络推广公司
  • 做网站领券收佣金/杭州seo公司哪家好
  • 网站设计制作哪种快/cba排名最新排名