基于NTP服务器获取网络时间的实现
文章目录
- 1 NTP
- 1.1 简介
- 1.2 包结构
- 1.3 UNIX 时间戳和NTP时间戳
- 2 代码实现
- 2.1 实现步骤
- 2.2 完整代码
- 3 结果
在某些场景下,单片机需要通过网络获取准确的时间进行数据同步,例如日志记录、定时任务等。然而,单片机本身无法直接获得准确的标准时间,往往需要依靠网络时间协议(NTP
)服务器来同步时间。本文将详细介绍如何通过NTP服务器获取准确的网络时间。
1 NTP
1.1 简介
NTP(Network Time Protocol
,网络时间协议)是一种用于使计算机时钟同步到互联网标准时间的协议。NTP服务器通常分层级(Stratum
)运作,一级服务器直接与时间基准同步,而其他级别的服务器则从更高级别的服务器获取时间。通过NTP,我们可以让系统的时钟保持与标准时间的一致性。
1.2 包结构
NTP包通常由以下字段组成:
li_vn_mode
:润秒指示器(2位)、版本号(3位)和模式(3位)。stratum
:服务器层级,1表示主服务器,2及更高层级表示从服务器,0表示未同步。poll
:连续NTP请求间的最大间隔,以2的幂次表示。precision
:服务器时钟的精度,以2的幂次表示。root_delay
:从NTP客户端到NTP服务器的根延迟(秒)。root_dispersion
:NTP服务器与其上一级时钟源的偏差(秒)。reference_identifier
:NTP服务器的参考标识符,通常指示服务器的源。reference_timestamp
:参考时间戳。origin_timestamp
:发起时间戳。receive_timestamp
:接收时间戳。transmit_timestamp
:传输时间戳。
1.3 UNIX 时间戳和NTP时间戳
- UNIX时间戳:指从1970年1月1日00:00:00 UTC开始到当前时刻的秒数。
- NTP时间戳:指从1900年1月1日00:00:00 UTC开始到当前时刻的秒数。
偏移量的计算:
- 因为 NTP 时间戳的起点比 UNIX 时间戳的起点早 70 年(从1900年到1970年)。
- 假设这 70 年内没有闰秒,70 年一共 25567 天,每天86400秒。因此,这个时间差的秒数为
70 * 365 + 17
天(17天是因为这段时间内包括了17 次闰年),每天86400秒。 - 所以这个偏移量可以计算为
2208988800
秒,即70 * 365 * 86400 + 17 * 86400
。
偏移量的用途:
- 在NTP协议中,时间戳以从1900年开始的秒数表示,而在许多其他系统中(如Unix系统),时间以从1970年开始的秒数表示。
- 因此,需要将NTP时间戳转换为UNIX时间戳,或者将UNIX时间戳转换为NTP时间戳,则需要用到上面的偏移。
2 代码实现
2.1 实现步骤
这里在Linux环境下为例对NTP时间进行获取,使用标准的POSIX/BSD
套接字编程,这样如果想在单片机中LwIP实现的话,也可以直接使用。
1、NTP服务端地址
这里我们以阿里云的NTP服务器ntp3.aliyun.com
为例,来获取获取网络时间,它的端口为123
。
#define NTP_SERVER "203.107.6.88" // NTP服务器地址:"ntp3.aliyun.com"
#define NTP_PORT 123 // NTP服务器端口号
- 本示例中不做DNS解析,我这里ping阿里的ntp服务器地址
ntp3.aliyun.com
的ip为203.107.6.88
。若用在程序中,建议做DNS解析。
2、NTP结构体
// NTP数据包结构体
typedef struct {uint8_t li_vn_mode; // 润秒指示器(2),版本号(3),模式(3)uint8_t stratum; // 服务器层级,1-主服务器,2-从服务器,0-未同步uint8_t poll; // 连续NTP请求间的最大间隔,2的幂表示uint8_t precision; // 服务器时钟的精度,2的幂表示(秒)uint32_t root_delay; // 从NTP客户端到NTP服务器的根延迟(秒)uint32_t root_dispersion; // NTP服务器与其上一级时钟源的偏差(秒)uint32_t reference_identifier; // NTP服务器的参考标识符,通常指示了服务器的源uint32_t reference_timestamp[2]; // 参考时间戳uint32_t origin_timestamp[2]; // 发起时间戳uint32_t receive_timestamp[2]; // 接收时间戳uint3232 transmit_timestamp[2]; // 传输时间戳
} ntp_packet;
2、创建套接字并设置超时时间
int sockfd;
struct timeval timeout;timeout.tv_sec = NTP_RCV_TIMEO / 1000;
timeout.tv_usec = (NTP_RCV_TIMEO % 1000) * 1000;// 创建socket
sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout));
这里创建套接字后,需要使用setsockopt
设置套接字的接收超时时间。这是因为NTP是UDP协议的,有可能请求了NTP报文后,请求报文没发送出去或响应报文没被正确接收,所以过了超时时间还没有接收到NTP时间的话,我们不能卡在读函数中。
3、设置服务端地址和NTP请求报文
#define NTP_UNIX_OFFSET 2208988800UL // UNIX时间戳(1970.1.1)和NTP时间戳(1900.1.1)之间的偏移量
struct sockaddr_in server_addr;
// 设置服务器地址
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(NTP_PORT);
server_addr.sin_addr.s_addr = inet_addr(NTP_SERVER);// 设置NTP版本和客户端模式
packet.li_vn_mode = (0x3 << 6) | (0x3 << 3) | 0x3; // NTP版本3、客户端模式
packet.transmit_timestamp[0] = htonl(NTP_UNIX_OFFSET); // 设置传输时间为偏移,则后面无需转换
在网络协议中,大多数字段都遵循“大端字节序”,即高位字节在前,低位字节在后。htonl
将主机字节序的 NTP_UNIX_OFFSET
转换为符合 NTP 协议要求的字节序。
4、请求、获取并输出NTP时间
// 请求NTP时间
sendto(sockfd, (const char*)&packet, sizeof(packet), 0, (struct sockaddr*)&server_addr, sizeof(server_addr));
recv_len = recvfrom(sockfd, (char*)&packet, sizeof(packet), 0, NULL, NULL);
// 成功接收,计算NTP时间
ntp_time = (time_t)(ntohl(packet.transmit_timestamp[0]) - NTP_UNIX_OFFSET);
printf("NTP time: %s", ctime(&ntp_time));
close(sockfd);
2.2 完整代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <time.h>
#include <unistd.h>#define NTP_SERVER "203.107.6.88" // NTP服务器地址:"ntp3.aliyun.com"
#define NTP_PORT 123 // NTP服务器端口号
#define NTP_PACKET_SIZE 48 // NTP数据包大小
#define NTP_UNIX_OFFSET 2208988800UL // UNIX时间戳(1970.1.1)和NTP时间戳(1900.1.1)之间的偏移量
#define NTP_RCV_TIMEOUT 2 // 接收超时时间(秒)
#define MAX_RETRIES 5 // 最大重试次数// NTP数据包结构体
typedef struct {uint8_t li_vn_mode; // 润秒指示器(2),版本号(3),模式(3)uint8_t stratum; // 服务器层级,1-主服务器,2-从服务器,0-未同步uint8_t poll; // 连续NTP请求间的最大间隔,2的幂表示uint8_t precision; // 服务器时钟的精度,2的幂表示(秒)uint32_t root_delay; // 从NTP客户端到NTP服务器的根延迟(秒)uint32_t root_dispersion; // NTP服务器与其上一级时钟源的偏差(秒)uint32_t reference_identifier; // NTP服务器的参考标识符,通常指示了服务器的源uint32_t reference_timestamp[2]; // 参考时间戳uint32_t origin_timestamp[2]; // 发起时间戳uint32_t receive_timestamp[2]; // 接收时间戳uint32_t transmit_timestamp[2]; // 传输时间戳
} ntp_packet;void ntp_get_time_once() {int sockfd;struct sockaddr_in server_addr;ntp_packet packet = {0};time_t ntp_time;int retries = 0;int recv_len;struct timeval timeout;timeout.tv_sec = NTP_RCV_TIMEOUT;timeout.tv_usec = 0;// 创建socketsockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);if (sockfd < 0) {perror("socket failed");return;}// 设置接收超时时间setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout));// 设置NTP服务器地址memset(&server_addr, 0, sizeof(server_addr));server_addr.sin_family = AF_INET;server_addr.sin_port = htons(NTP_PORT);server_addr.sin_addr.s_addr = inet_addr(NTP_SERVER);// 初始化NTP请求数据包packet.li_vn_mode = (0x3 << 6) | (0x3 << 3) | 0x3; // NTP版本3、客户端模式packet.transmit_timestamp[0] = htonl(NTP_UNIX_OFFSET);// 重试逻辑while (retries < MAX_RETRIES) {// 发送NTP请求if (sendto(sockfd, (const char*)&packet, sizeof(packet), 0, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0) {perror("sendto failed");close(sockfd);return;}// 接收NTP响应recv_len = recvfrom(sockfd, (char*)&packet, sizeof(packet), 0, NULL, NULL);if (recv_len >= 0) {// 成功接收,计算NTP时间ntp_time = (time_t)(ntohl(packet.transmit_timestamp[0]) - NTP_UNIX_OFFSET);printf("NTP time: %s", ctime(&ntp_time));close(sockfd);return;} else {perror("recvfrom failed, retrying...");retries++;}}// 达到最大重试次数后失败printf("NTP sync failed after %d retries.\n", MAX_RETRIES);close(sockfd);
}int main() {ntp_get_time_once();return 0;
}
我这里代码就是简单地重新请求固定次数后就退出。大家可以根据实际情况进行更改,比如一定要获取到了再返回。
3 结果
编译上面的代码并运行,可以看到输出了当前的时间:
这里阿里的NTP服务器返回的是标准的UTC时间,加上12小时为当前的北京时间。
相关文章:

基于NTP服务器获取网络时间的实现
文章目录 1 NTP1.1 简介1.2 包结构1.3 UNIX 时间戳和NTP时间戳 2 代码实现2.1 实现步骤2.2 完整代码 3 结果 在某些场景下,单片机需要通过网络获取准确的时间进行数据同步,例如日志记录、定时任务等。然而,单片机本身无法直接获得准确的标准时…...

Web APIs(获取元素+操作元素+节点操作)
目录 1.API 和 Web API 2.DOM导读 DOM树 3.获取元素 getElementById获取元素 getElementsByTagName获取元素 H5新增方法获取 获取特殊元素 4.事件基础 执行事件 操作元素 修改表单属性 修改样式属性 使用className修改样式属性 获取属性的值 设置属性的值 移除…...
Android adb shell关于CPU核的命令
Android adb shell关于CPU核的命令 先使用命令: adb shell 进入控制台。 然后,直接在$后面输入下面命令,针对CPU的命令。 cat /proc/cpuinfo | grep ^processor | wc -l 查看当前手机的CPU是几核的。 cat sys/devices/system/cpu/online …...

基于springboot+mybatis+vue的项目实战之页面参数传递
如图所示,删除操作可以用按钮实现,也可以用超链接来实现。 1、第一种情况,用按钮实现。 html页面相关: <button type"button" click"deleteId(peot.id)">删除</button> <script>new Vue(…...

CSS-浮动
float (浮动) 作用:盒子的顶点是一样的,具备行内块的特征,能设置宽高 属性:float 属性值:left 浮动在网页左边 right 浮动在网页右边 .a{width: 100px;height: 100px;float:left;background-color: red;}.b…...
MFC:字符串处理
例子 //多字节char* szTest "abc多字节";int nLen strlen(szTest);//9//宽字节wchar_t* szTest2 L"abc多字节";int nlen2 wcslen(szTest2);//6//测试项目配置为Unicodewchar_t* szTesz3 TEXT("abcd");//char* -> CStringCString strTes…...

虚拟仿真云平台在教育应用中的优势和意义
虚拟仿真云实验教学平台作为一种新型的教学方法,近年来在高校教育中得到了十分广泛的应用。它通过模拟真实的实验场景和实验操作,让学生在计算机上进行实验操作和数据处理,为学生提供了更加便捷、可靠、有效的实验学习环境。本文,…...
CPU的的处理流程如何快速记忆
为了快速记忆CPU的处理流程,可以将其简化成五个主要阶段,通常称为“冯诺依曼架构”的五个基本步骤,或者是流水线处理的几个阶段。下面是一种便于记忆的简化版本: CPU处理流程的五个阶段: 取指令(Instructi…...

AI视频教程下载:基于OpenAl、LangChain、 Replicate开发AI应用程序
欢迎来到令人兴奋的 AI 应用世界!在这门课程中,你将学习到创建一个能够与用户互动、理解自然语言、处理音频输入,甚至分析图像的真正智能应用所需的技能和技术。 AI 工具和技术 你将获得使用几个知名 AI API 和技术的实际经验。这些行业领先…...

【C++】继承相关(基类与派生类的继承关系以及细节整理)
目录 00.引言 01.继承的定义 02.基类和派生类对象 03.继承中的作用域 04.派生类的默认成员函数 05.友元、静态成员 00.引言 继承是面向对象编程中的一个重要概念,它的作用是创建一个新的类,该类可以从一个已存在的类(父类/基类&#x…...
【Web后端】监听器Listener
1、简介 用来监听Servlet组件对象状态发生变化的组件可以监听的源包括:ServetRequest、HttpSession、ServletContext当监听到事件源状态发生变化时,会有对应的响应行为 2、使用方法 在web.xml文件中配置 <listener> <listener-class>com.coder.util.…...

C/C++ 初级球球大作战练手
效果演示: https://live.csdn.net/v/385490 游戏初始化 #include <stdbool.h> #include<stdio.h> #include<stdlib.h> #include<time.h> #include<graphics.h> #include <algorithm> #include<math.h> #include<mmsy…...
ES6之字符串的扩展
字符串的扩展 关键的扩展点及其示例: Unicode 表示与处理 JavaScript 共有6种方法可以表示一个字符。codePointAtpos:String.fromCodePoint…codePoints: **字符串的遍历 for … of **字符串方法的增强 includessearchString[, position]&…...
微信小程序开发,构建NPM报错解决
报错信息如下: message: NPM packages not found. Please confirm npm packages which need to build are belong to miniprogramRoot directory. Or you may edit project.config.jsons packNpmManually and packNpmRelationList appid: wx7144a5d5411…...

【异常】SpringBoot整合RabbitMQ-发送消息报错
错误信息 reply-code406, reply-textPRECONDITION_FAILED - inequivalent arg ‘x-message-ttl’ for queue ‘hello-queue’ in vhost ‘/lq’: received none but current is the value ‘10000’ of type ‘signedint’, class-id50, method-id10 错误原因 hello-queue这…...

通过钉钉卡片进行工单审批
我们通常通过钉钉机器人来发送通知,提醒审批人名下有待办工单需要处理。这种通知方式仅能提醒审批人到ITSM中处理,审批人需要打开电脑登陆平台处理,我们就考虑是否能有一种方式能够满足移动端审批? 这里我们可以使用ITSM的移动端版…...
C语言中的控制语句:深入解析与案例实践2
五、使用goto实现无条件跳转 虽然goto语句在现代编程实践中往往被避免使用,因为它可能导致代码难以理解和维护,但在某些特定情况下,它仍然可以作为一种解决方案。goto语句允许程序无条件地跳转到代码中指定的标签位置。 案例代码:…...

【ARM Cortex-M 系列 2.1 -- Cortex-M7 Debug system registers】
请阅读【嵌入式开发学习必备专栏】 文章目录 Debug system registers中断控制状态寄存器(ICSR)Debug Halting Control and Status Register, DHCSR Debug 寄存器DCRSR与DCRDRCPU 寄存器读操作CPU 寄存器写操作CPU 寄存器选择CPU 寄存器读写示例 调试故障…...

深入解析C#中的async和await关键字
文章目录 一、异步编程的基本概念及其在C#中的实现二、async关键字的定义及其用法三、await关键字的定义及其用法示例代码:使用async和await编写一个简单的异步程序 四、async和await的优点注意事项 五、C#下async和await中常见问题汇总1. 异步方法中的await调用2. …...

【VUE.js】前端框架——未完成
基于脚手架创建前端工程 环境 当安装node.js时,它本身就携带有npm命令。(-v 查版本号)安装VUE CLI npm i vue/cli -g(全局) 创建 vue create 【project name】 镜像源解决方案 输入创建命令后,提示检查更…...

label-studio的使用教程(导入本地路径)
文章目录 1. 准备环境2. 脚本启动2.1 Windows2.2 Linux 3. 安装label-studio机器学习后端3.1 pip安装(推荐)3.2 GitHub仓库安装 4. 后端配置4.1 yolo环境4.2 引入后端模型4.3 修改脚本4.4 启动后端 5. 标注工程5.1 创建工程5.2 配置图片路径5.3 配置工程类型标签5.4 配置模型5.…...

VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
Java如何权衡是使用无序的数组还是有序的数组
在 Java 中,选择有序数组还是无序数组取决于具体场景的性能需求与操作特点。以下是关键权衡因素及决策指南: ⚖️ 核心权衡维度 维度有序数组无序数组查询性能二分查找 O(log n) ✅线性扫描 O(n) ❌插入/删除需移位维护顺序 O(n) ❌直接操作尾部 O(1) ✅内存开销与无序数组相…...
vue3 定时器-定义全局方法 vue+ts
1.创建ts文件 路径:src/utils/timer.ts 完整代码: import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...
AI编程--插件对比分析:CodeRider、GitHub Copilot及其他
AI编程插件对比分析:CodeRider、GitHub Copilot及其他 随着人工智能技术的快速发展,AI编程插件已成为提升开发者生产力的重要工具。CodeRider和GitHub Copilot作为市场上的领先者,分别以其独特的特性和生态系统吸引了大量开发者。本文将从功…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...

学习STC51单片机32(芯片为STC89C52RCRC)OLED显示屏2
每日一言 今天的每一份坚持,都是在为未来积攒底气。 案例:OLED显示一个A 这边观察到一个点,怎么雪花了就是都是乱七八糟的占满了屏幕。。 解释 : 如果代码里信号切换太快(比如 SDA 刚变,SCL 立刻变&#…...

什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
Device Mapper 机制
Device Mapper 机制详解 Device Mapper(简称 DM)是 Linux 内核中的一套通用块设备映射框架,为 LVM、加密磁盘、RAID 等提供底层支持。本文将详细介绍 Device Mapper 的原理、实现、内核配置、常用工具、操作测试流程,并配以详细的…...
Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析
Java求职者面试指南:Spring、Spring Boot、MyBatis框架与计算机基础问题解析 一、第一轮提问(基础概念问题) 1. 请解释Spring框架的核心容器是什么?它在Spring中起到什么作用? Spring框架的核心容器是IoC容器&#…...