STM32——RTC实时时钟(时间戳)

目录

一、Unix时间戳

1、概念

2、时间戳使用

二、RTC实时时钟

1、介绍

2、代码


一、Unix时间戳

1、概念

Unix 时间戳(Unix Timestamp)定义为从UTC/GMT的 1970年1月1日 0时0分0秒 开始所经过的秒数,不考虑闰秒。

  • 时间戳存储在一个秒计数器中,秒计数器为32位/64位的整型变量
  • 世界上所有时区的秒计数器相同,不同时区通过添加偏移来得到当地时间

2、时间戳使用

主要函数,使用时要包含C语言的 "time.h" 模块

参数:

time_t 类型:32/64位秒数。        struct tm* 类型:指向日期结构体的指针

struct tm {
   int tm_sec;         /* 秒,范围从 0 到 59		*/
   int tm_min;         /* 分,范围从 0 到 59		*/
   int tm_hour;        /* 小时,范围从 0 到 23		*/
   int tm_mday;        /* 一月中的第几天,范围从 1 到 31	*/
   int tm_mon;         /* 月,范围从 0 到 11		*/
   int tm_year;        /* 自 1900 年起的年数		*/
   int tm_wday;        /* 一周中的第几天,范围从 0 到 6	*/
   int tm_yday;        /* 一年中的第几天,范围从 0 到 365	*/
   int tm_isdst;       /* 夏令时				*/
};

使用样例: 

///秒计数器类型
time_t time_cnt = 1682953445;        //浏览器搜索时间戳在线转换工具
time_t time_rcv = 0;

///日期类型
struct tm* time_date;

///字符串类型
char * time_str;

void time_test(void)
{
        time_date = localtime( &time_cnt );
        printf("%d年-%d月-%d日--%d时-%d分-%d秒\r\n",
               time_date->tm_year+1900, time_date->tm_mon+1, time_date->tm_mday,
               time_date->tm_hour, time_date->tm_min, time_date->tm_sec          );
        
        time_rcv = mktime( time_date );
        printf("%d\r\n", time_rcv);
        
        time_str = ctime( &time_cnt );
        printf("秒数字符串:%s", time_str);
        
        time_str = asctime( time_date );
        printf("日期字符串:%s\r\n", time_str);
}
/*串口打印
2023年-5月-1日--15时-4分-5秒
1682953445
秒数字符串:Mon May  1 15:04:05 2023
日期字符串:Mon May  1 15:04:05 2023
*/

二、RTC实时时钟

1、介绍

RTC(Real Time Clock)实时时钟,本质是一个定时器

RTC是一个独立的定时器,可为系统提供时钟和日历的功能 RTC和时钟配置系统处于后备区域,系统复位时数据不清零,VDD(2.0~3.6V)断电后可借助VBAT(1.8~3.6V)供电继续走时

32位的可编程计数器,可对应Unix时间戳的秒计数器 20位的可编程预分频器,可适配不同频率的输入时钟,可选择三种RTC时钟源:     

  • HSE时钟除以128(通常为8MHz/128)     
  • LSE振荡器时钟(通常为32.768KHz)     
  • LSI振荡器时钟(40KHz)

框架图:

江科大:

RTC操作注意事项:

执行以下操作将使能对BKP和RTC的访问:
	设置RCC_APB1ENR的PWREN和BKPEN,使能PWR和BKP时钟
	设置PWR_CR的DBP,使能对BKP和RTC的访问

若在读取RTC寄存器时,RTC的APB1接口曾经处于禁止状态,
则软件首先必须等待RTC_CRL寄存器中的RSF位(寄存器同步标志)被硬件置1

必须设置RTC_CRL寄存器中的CNF位,使RTC进入配置模式后,
才能写入RTC_PRL、RTC_CNT、RTC_ALR寄存器

对RTC任何寄存器的写操作,都必须在前一次写操作结束后进行。
可以通过查询RTC_CR寄存器中的RTOFF状态位,判断RTC寄存器是否处于更新中。
仅当RTOFF状态位是1时,才可以写入RTC寄存器

2、代码

//初始化配置
void MyRTC_Init(void)
{
	/*开启时钟*/
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE);		//开启PWR的时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_BKP, ENABLE);		//开启BKP的时钟
	
	/*备份寄存器访问使能*/
	PWR_BackupAccessCmd(ENABLE);				      //使用PWR开启对备份寄存器的访问
	
    //通过写入备份寄存器的标志位,判断RTC是否是第一次配置
	if (BKP_ReadBackupRegister(BKP_DR1) != 0xA5A5)
	{
		RCC_LSEConfig(RCC_LSE_ON);							//开启LSE时钟
		while (RCC_GetFlagStatus(RCC_FLAG_LSERDY) != SET);	//等待LSE准备就绪
		
        
		RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);				//选择RTCCLK来源为LSE
		RCC_RTCCLKCmd(ENABLE);								//RTCCLK使能
		
		RTC_WaitForSynchro();								//等待同步
		RTC_WaitForLastTask();								//等待上一次操作完成
		
		RTC_SetPrescaler(32768 - 1);	//设置RTC预分频器,预分频后的计数频率为1Hz
		RTC_WaitForLastTask();			//等待上一次操作完成
		
		MyRTC_SetTime();	//设置时间,调用此函数,全局数组里时间值刷新到RTC硬件电路
		
        //在备份寄存器写入自己规定的标志位,用于判断RTC是不是第一次执行配置
		BKP_WriteBackupRegister(BKP_DR1, 0xA5A5);
	}
	else													//RTC不是第一次配置
	{
		RTC_WaitForSynchro();								//等待同步
		RTC_WaitForLastTask();								//等待上一次操作完成
	}
}

//设置时间
uint16_t MyRTC_Time[] = {2023, 1, 1, 23, 59, 55};
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(Real-Time Clock)的时钟设置或配置,需按照特定流程完成一系列操作。以下是关于如何调整RTC时钟及相关参数的具体说明。 #### 1. 配置RTC时钟源 在启用RTC之前,必须先选择合适的时钟源并使其生效。通常可以选择LSI、LSE或者HSE作为RTC时钟源。如果使用外部晶振(LSE),则需要通过函数`RCC_LSEConfig()`开启LSE功能[^1]: ```c // 启用LSE RCC_LSEConfig(RCC_LSE_ON); ``` 随后调用`RCC_RTCCLKConfig()`来指定RTC使用的具体时钟源,并通过`RCC_RTCCLKCmd()`激活RTC时钟信号: ```c // 设置RTC时钟源为LSE RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE); // 开启RTC时钟 RCC_RTCCLKCmd(ENABLE); ``` #### 2. 初始化RTC模块 初始化过程涉及设定分频系数以及定义时间格式等内容。这一步骤由`RTC_Init()`实现,其中包含了同步预分频器和异步预分频器的配置选项[^2]。例如: ```c RTC_InitTypeDef RTC_InitStructure; // 填充结构体成员变量 RTC_InitStructure.RTC_HourFormat = RTC_HourFormat_24; RTC_InitStructure.RTC_AsynchPrediv = 99; // 异步分频值 (取决于所选时钟源频率) RTC_InitStructure.RTC_SynchPrediv = 255; // 同步分频值 // 调用初始化API RTC_Init(&RTC_InitStructure); ``` #### 3. 更新时间和日期 一旦完成了上述基础性的准备工作之后,就可以利用专门接口分别写入新的时间与日期数据到对应的寄存器里去: ```c RTC_TimeTypeDef RTC_TimeStructure; RTC_DateTypeDef RTC_DateStructure; /* 设定时间为:HH=18 MM=30 SS=0 */ RTC_TimeStructure.RTC_Hours = 18; RTC_TimeStructure.RTC_Minutes = 30; RTC_TimeStructure.RTC_Seconds = 0; RTC_SetTime(RTC_Format_BIN, &RTC_TimeStructure); /* 设定日期为:DD=15 MO=7 YEAR=23 */ RTC_DateStructure.RTC_WeekDay = RTC_Thursday; RTC_DateStructure.RTC_Month = RTC_Month_July; RTC_DateStructure.RTC_Date = 15; RTC_DateStructure.RTC-Year = 23; RTC_SetDate(RTC_Format_BIN, &RTC_DateStructure); ``` 以上代码片段展示了怎样把当前时刻设成下午六点半整;同理也演示了将某一天标记为当年七月十五号星期四的方法。 #### 注意事项 - **电源管理**:由于RTC依赖于Vbat供电维持运行状态,在执行任何更改前应确认已允许访问备份区域内的资源。 ```c PWR_BackupAccessCmd(ENABLE); ``` - **锁机制**:某些版本可能还需要解锁保护位才能进一步操作相关寄存器[^2]。 --- ###
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值