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

STM32入门——实时时钟(20)

STM32入门——实时时钟(20)

RTC简介

  • RTC(Real Time Clock)实时时钟

  • RTC是一个独立的定时器,可为系统提供时钟和日历的功能

  • RTC和时钟配置系统处于后备区域,系统复位时数据不清零,VDD(2.03.6V)断电后可借助VBAT(1.8V3.6V)供电继续走时。

  • 32位的可编程计数器,可对应Unix时间戳的秒计数器。

  • 20位的可编程预分频器,可适配不同频率的输入时钟。

  • 可选择三种RTC时钟源:
    HSE时钟除以128(通常为8MHz/128)

    LSE振荡器时钟(通常为32.768KHz)

    LSI振荡器时钟(40KHz)

image-20260405180507651

image-20260405180530919

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyRTC.h"int main(void)
{/*模块初始化*/OLED_Init();		//OLED初始化MyRTC_Init();		//RTC初始化/*显示静态字符串*/OLED_ShowString(1, 1, "Date:XXXX-XX-XX");OLED_ShowString(2, 1, "Time:XX:XX:XX");OLED_ShowString(3, 1, "CNT :");OLED_ShowString(4, 1, "DIV :");while (1){MyRTC_ReadTime();							//RTC读取时间,最新的时间存储到MyRTC_Time数组中OLED_ShowNum(1, 6, MyRTC_Time[0], 4);		//显示MyRTC_Time数组中的时间值,年OLED_ShowNum(1, 11, MyRTC_Time[1], 2);		//月OLED_ShowNum(1, 14, MyRTC_Time[2], 2);		//日OLED_ShowNum(2, 6, MyRTC_Time[3], 2);		//时OLED_ShowNum(2, 9, MyRTC_Time[4], 2);		//分OLED_ShowNum(2, 12, MyRTC_Time[5], 2);		//秒OLED_ShowNum(3, 6, RTC_GetCounter(), 10);	//显示32位的秒计数器OLED_ShowNum(4, 6, RTC_GetDivider(), 10);	//显示余数寄存器}
}
#include "stm32f10x.h"
#include "time.h"uint16_t MyRTC_Time[] = {2026,1,1,23,59,55};      // 定义全局的时间数组,数组内容分别为年、月、日、时、分、秒void MyRTC_SetTime(void);                        // 函数声明/*** 函    数:RTC初始化* 参    数:无* 返 回 值:无*/
void MyRTC_Init(void)
{/*开启时钟*/RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);         // 开启PWR的时钟RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP,ENABLE);         // 开启BKP的时钟/*备份寄存器访问使能*/PWR_BackupAccessCmd(ENABLE);                               // 使用PWR开启对备份寄存器的访问if(BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)              // 通过写入备份寄存器的标志位,判断RTC是否是第一次配置// if成立则执行第一次的RTC配置{RCC_LSEConfig(RCC_LSE_ON);                             // 开启LSE时钟while(RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET);      // 等待LSE准备就绪RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);                // 选择RTCCLK来源为LSERCC_RTCCLKCmd(ENABLE);                                 // RTCCLK使能RTC_WaitForSynchro();                                  // 等待同步RTC_WaitForLastTask();                                 // 等待上一次操作完成RTC_SetPrescaler(32768 - 1);                           // 设置RTC预分频器,预分频后的计数频率为1HzRTC_WaitForLastTask();                                 // 等待上一次操作完成MyRTC_SetTime();                                       // 设置时间,调用此函数,全局数组里时间值刷新到RTC硬件电路BKP_WriteBackupRegister(BKP_DR1,0xA5A5);               // 在备份寄存器写入自己规定的标志位,用于判断RTC是不是第一次执行配置}else                                                       // RTC不是第一次配置{RTC_WaitForSynchro();                                  // 等待同步RTC_WaitForLastTask();                                 // 等待上一次操作完成}
}// 如果LSE无法起振导致程序卡死在初始化函数中
// 可将初始化函数替换为下述代码,使用LSI当作RTCCLK
// LSI无法由备用电池供电,故主电源掉电后,RTC走时会暂停
/*
void MyRTC_Init(void)
{RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP,ENABLE);PWR_BackupAccessCmd(ENABLE);if(BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5){RCC_LSICmd(ENABLE);while(RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET);RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);RCC_RTCCLKCmd(ENABLE);RTC_WaitForSynchro();RTC_WaitForLastTask();RTC_SetPrescaler(40000 - 1);RTC_WaitForLastTask();MyRTC_SetTime();BKP_WriteBackupRegister(BKP_DR1,0xA5A5);}else{RCC_LSICmd(ENABLE);                                    // 即使不是第一次配置,也需要再次开启LSI时钟while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY) != SET);RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);RCC_RTCCLKCmd(ENABLE);RTC_WaitForSynchro();RTC_WaitForLastTask();}
}
*//*** 函    数:RTC设置时间* 参    数:无* 返 回 值:无* 说    明:调用此函数后,全局数组里时间值将刷新到RTC硬件电路*/
void MyRTC_SetTime(void)
{time_t Time_cnt;                 // 定义秒计数器数据类型struct tm time_date;             // 定义日期时间数据类型time_date.tm_year = MyRTC_Time[0] - 1900;        // 将数组的时间赋值给日期时间结构体time_date.tm_mon = MyRTC_Time[1] - 1;time_date.tm_mday = MyRTC_Time[2];time_date.tm_hour = MyRTC_Time[3];time_date.tm_min = MyRTC_Time[4];time_date.tm_sec = MyRTC_Time[5];Time_cnt = mktime(&time_date) - 8 * 60 * 60;    // 调用mktime函数,将日期时间转换为秒计数器格式// - 8 * 60 * 60 为东八区的时区调整RTC_SetCounter(Time_cnt);                       // 将秒计数器写入到RTC的CNT中RTC_WaitForLastTask();                          // 等待上一次操作完成
}/*** 函    数:RTC读取时间* 参    数:无* 返 回 值:无* 说    明:调用此函数后,RTC硬件电路里时间值将刷新到全局数组*/
void MyRTC_ReadTime(void)
{time_t time_cnt;                   // 定义秒计数器数据类型struct tm time_date;               // 定义日期时间数据类型time_cnt = RTC_GetCounter() + 8 * 60 * 60;            // 读取RTC的CNT,获取当前的秒计数器// + 8 * 8 * 60为东八区的时区调整time_date = *localtime(&time_cnt);                   // 将localtime函数,将秒计数器转换为日期时间格式MyRTC_Time[0] = time_date.tm_year + 1900;		     //将日期时间结构体赋值给数组的时间MyRTC_Time[1] = time_date.tm_mon + 1;MyRTC_Time[2] = time_date.tm_mday;MyRTC_Time[3] = time_date.tm_hour;MyRTC_Time[4] = time_date.tm_min;MyRTC_Time[5] = time_date.tm_sec;}
#ifndef __MYRTC_H
#define __MYRTC_Hextern uint16_t MyRTC_Time[];void MyRTC_Init(void);
void MyRTC_SetTime(void);
void MyRTC_ReadTime(void);#endif
http://www.jsqmd.com/news/593165/

相关文章:

  • 用Unity 2D碰撞体+Effector,5分钟实现《星露谷物语》式的磁铁吸附效果
  • Kotlin的泛型上下界,枚举 和 字符串资源的区别
  • 告别手动调参:利用快马ai批量对比优化openclaw配置效率提升十倍
  • Roots and the Stars[1]
  • ai辅助开发新体验:让快马ai帮你打造智能win10安装准备助手
  • 降AI前需要做哪些准备:规范操作流程和自查清单 - 还在做实验的师兄
  • 利用快马平台快速原型设计,模拟智能应用控制拦截提示界面
  • 价值:社会对劳动所产生的效用增量形成的局部共识
  • 基于微电网的小信号建模下垂控制稳定性的根轨迹分析
  • MySQL 8.0 与 5.7 全维度深度对比:核心差异、技术演进与新手选型终极指南
  • 基于.NET的Windows窗体编程之WinForms布局简介
  • 云边端数据一致性:分布式事务的实现方法
  • 降AI后怎么验证效果:多平台交叉检测的完整操作方法 - 还在做实验的师兄
  • 告别重复劳动:用快马生成智能安装脚本,一键统一团队openclaw环境配置
  • D8 209.字母异位词 349.两个数组的交集
  • python tarfile
  • AI时代的价值冲击——共识瓦解与转型阵痛
  • 赋能crm新智能:利用快马ai模型实现客户分析与话术建议
  • 降AI工具处理理科公式和图表对比:哪款保留效果最好 - 还在做实验的师兄
  • ESP32 YoRadio架构深度解析:开源网络收音机实现原理与技术配置指南
  • 直流电机双闭环调速控制系统仿真模型 转速电流双闭环PI控制 Matlab/Simulink仿真模型 带报告
  • DisplayLink驱动在Linux系统上的跨发行版兼容性解决方案
  • League Akari 技术解析:从算法原理到实战应用
  • 新手零压力上手:在快马平台生成你的第一个模拟xshell终端
  • 如何用嘎嘎降AI处理文献综述部分:综述专项降AI教程 - 还在做实验的师兄
  • 告别混乱日志:用Logrus Hook为你的Gin应用打造可观测性系统
  • 5G信号总断?可能是瑞利衰落在捣鬼!手把手教你用MATLAB仿真分析
  • 零基础掌握AI动作迁移:ComfyUI-MimicMotionWrapper全攻略
  • BetterJoy终极指南:在Windows电脑上完美使用Switch手柄玩游戏
  • Mac光标个性化新纪元:Mousecape让你的指针体验全面升级