前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >ESP8266调用NTP服务器进行时间校准

ESP8266调用NTP服务器进行时间校准

作者头像
DS小龙哥
发布2023-07-08 20:04:32
1K0
发布2023-07-08 20:04:32
举报
文章被收录于专栏:嵌入式项目开发

一、背景知识

【1】什么是NTP服务器?

NTP是网络时间协议(Network Time Protocol,简称NTP),是一种用于同步计算机时间的协议。NTP服务器指的是提供NTP服务的计算机或设备。NTP服务器的主要功能是保证网络上的所有设备的时间同步,以确保各个设备相互之间的时间协调一致。NTP服务器通常连接到具有高度精确时间源的设备,例如:GPS接收器或原子钟,以确保提供准确如一的时间。网络上的计算机可以通过连接到NTP服务器来同步其时间,并确保它们在同一时刻进行操作。

img
img

目前有许多可以使用的NTP服务器,以下是一些常用的NTP服务器列表:

代码语言:javascript
复制
1. cn.ntp.org.cn
2. ntp.sjtu.edu.cn
3. ntp.linux.org.cn
4. time.nist.gov.cn
5. ntp.aliyun.com
6. ntp.api.bz
7. ntp1.aliyun.com
8. time1.cloud.tencent.com
9. pool.ntp.org.cn

【2】RTC实时时钟是什么?

RTC (Real-Time Clock)实时时钟,是指一种专门用于记忆日期、时间的计时芯片或模块。一般包括一个时钟芯片、一块石英晶体、一块温度补偿电路、电源管理电路等组成。RTC可以精确地记录日期和时间,即使是在断电等异常情况下,也能保持记录的时间长达数年。常常用于嵌入式系统、数据采集设备等领域,是一种至关重要的设备。在某些系统应用中,RTC也会成为其他设备的时钟源,如单片机或微控制器单位等。

RTC的时间精度通常为ppm 级别,即每百万分之一,能够满足大多数实时应用场景的要求。为了提高RTC的稳定度和精度,许多RTC都带有自动校正功能,可以自动从外部时钟源或NTP服务器中获取准确的时间,并进行校正。同时,许多RTC还会集成电源管理功能,支持低功耗模式以延长电池寿命。

二、ESP8266获取网络时间

要通过ESP8266联网并获取网络时间,需要执行以下步骤:

  1. 在STM32F103ZET6上配置UART串口以与ESP8266进行通信。
  2. 使用AT指令将ESP8266连接到Wi-Fi网络。可以使用以下指令:
代码语言:javascript
复制
AT+CWJAP="SSID","password"

其中,替换 “SSID” 为自己的Wi-Fi网络名称,“password” 是Wi-Fi密码。

  1. 使用AT指令连接到NTP服务器并获取时间。您可以使用以下指令:
代码语言:javascript
复制
AT+CIPSNTPCFG=0,1,"pool.ntp.org"
AT+CIPSNTPTIME?

这将连接到ntp服务器并检索当前的UTC时间。

  1. 将ESP8266返回的UTC时间转换为本地时间。您需要知道您所在的时区,并对UTC进行适当的调整。
  2. 将本地时间设置为STM32F103ZET6上的RTC实时时钟。

下面是一个示例代码

代码语言:javascript
复制
#include <stdio.h>
#include "stm32f10x.h"

// UART配置
void uart_init() {
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);

    USART_InitTypeDef USART_InitStructure;
    USART_InitStructure.USART_BaudRate = 115200;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    USART_Init(USART1, &USART_InitStructure);

    USART_Cmd(USART1, ENABLE);
}

// 发送AT指令并等待响应
int send_at_command(char* command, char* response, uint32_t timeout) {
  // 发送命令
  USART_SendData(USART1, (uint8_t*)command, strlen(command));
  
  // 等待响应
  uint32_t start_time = HAL_GetTick();
  while ((HAL_GetTick() - start_time) < timeout) {
    if (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) != RESET) {
      char c = USART_ReceiveData(USART1);
      
      // 检查是否收到了预期的响应
      if (strstr(response, c) != NULL) {
        return 0; // 成功
      }
    }
  }
  
  return -1; // 超时或没有收到预期的响应
}

// 连接ESP8266到Wi-Fi
void connect_to_wifi() {
  char command[50];
  char response[100];
  
  // 设置Wi-Fi SSID和密码
  sprintf(command, "AT+CWJAP=\"%s\",\"%s\"\r\n", "YourSSID", "YourPassword");
  send_at_command(command, "OK", 5000);
}

// 连接到NTP服务器并获取时间
int get_ntp_time(uint32_t* time) {
  char response[100];
  
  // 配置SNTP客户端
  send_at_command("AT+CIPSNTPCFG=0,1,\"pool.ntp.org\"\r\n", "OK", 5000);
  
  // 获取时间
  send_at_command("AT+CIPSNTPTIME?\r\n", response, 5000);
  
  // 解析响应并提取时间戳
  char* token = strtok(response, ",");
  uint32_t timestamp = atoi(token);
  *time = timestamp - 2208988800UL; // 转换为Unix时间戳
  
  return 0;
}

// 将时间设置到RTC
void set_rtc_time(uint32_t time) {
  // 启用PWR和BKP外设时钟
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);
  
  // 解锁备份寄存器区域
  PWR_BackupAccessCmd(ENABLE);
  
  // 配置RTC
  RCC_RTCCLKConfig(RCC_RTCCLKSource_HSE_Div128); // RTC时钟源为HSE/128
  RCC_RTCCLKCmd(ENABLE); // 启用RTC时钟
  
  RTC_InitTypeDef RTC_InitStructure;
  // 配置RTC时钟 
    RTC_InitStructure.RTC_HourFormat = RTC_HourFormat_24; RTC_InitStructure.RTC_AsynchPrediv = 127; 
    RTC_InitStructure.RTC_SynchPrediv = 255; 
    RTC_Init(&RTC_InitStructure);

// 设置RTC时间 
    RTC_TimeTypeDef RTC_TimeStruct; 
    RTC_DateTypeDef RTC_DateStruct;

// 将Unix时间戳转换为RTC时间和日期 
  uint32_t days = time / 86400; 
    uint32_t seconds = time % 86400; 
    uint32_t hours = seconds / 3600; 
    uint32_t minutes = (seconds % 3600) / 60; 
    uint32_t secs = (seconds % 3600) % 60; 
    uint32_t year = 1970; 
    uint32_t month = 1; 
    while (days > 365) { if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0) { days -= 366; } else { days -= 365; } year++; } 
    while (days > 0) { if (month == 2) 
    { if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0) { if (days > 29) { days -= 29; } else { break; } } else { if (days > 28) { days -= 28; } else { break; } } } else if (month == 4 || month == 6 || month == 9 || month == 11) { if (days > 30) { days -= 30; } else { break; } } else { if (days > 31) { days -= 31; } else { break; } } month++; if (month > 12) { month = 1; year++; } }

RTC_TimeStruct.RTC_Hours = hours; RTC_TimeStruct.RTC_Minutes = minutes; RTC_TimeStruct.RTC_Seconds = secs; RTC_DateStruct.RTC_Date = days; RTC_DateStruct.RTC_Month = month; RTC_DateStruct.RTC_Year = year - 2000;

// 设置RTC时间和日期 
    RTC_SetTime(RTC_Format_BIN, &RTC_TimeStruct); 
    RTC_SetDate(RTC_Format_BIN, &RTC_DateStruct); }

int main() 
{ 
    // 初始化UART串口 
    uart_init();

   // 连接ESP8266到Wi-Fi
    connect_to_wifi();

// 获取NTP时间 
    uint32_t ntp_time; get_ntp_time(&ntp_time);

// 将时间设置到
    RTC set_rtc_time(ntp_time);

while (1) { // 做其他的事情... } }
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-06-07,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、背景知识
    • 【1】什么是NTP服务器?
      • 【2】RTC实时时钟是什么?
      • 二、ESP8266获取网络时间
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档