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

基于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 结果 在某些场景下&#xff0c;单片机需要通过网络获取准确的时间进行数据同步&#xff0c;例如日志记录、定时任务等。然而&#xff0c;单片机本身无法直接获得准确的标准时…...

Web APIs(获取元素+操作元素+节点操作)

目录 1.API 和 Web API 2.DOM导读 DOM树 3.获取元素 getElementById获取元素 getElementsByTagName获取元素 H5新增方法获取 获取特殊元素 4.事件基础 执行事件 操作元素 修改表单属性 修改样式属性 使用className修改样式属性 获取属性的值 设置属性的值 移除…...

Android adb shell关于CPU核的命令

Android adb shell关于CPU核的命令 先使用命令&#xff1a; adb shell 进入控制台。 然后&#xff0c;直接在$后面输入下面命令&#xff0c;针对CPU的命令。 cat /proc/cpuinfo | grep ^processor | wc -l 查看当前手机的CPU是几核的。 cat sys/devices/system/cpu/online …...

基于springboot+mybatis+vue的项目实战之页面参数传递

如图所示&#xff0c;删除操作可以用按钮实现&#xff0c;也可以用超链接来实现。 1、第一种情况&#xff0c;用按钮实现。 html页面相关&#xff1a; <button type"button" click"deleteId(peot.id)">删除</button> <script>new Vue(…...

CSS-浮动

float (浮动) 作用&#xff1a;盒子的顶点是一样的&#xff0c;具备行内块的特征&#xff0c;能设置宽高 属性&#xff1a;float 属性值&#xff1a;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…...

虚拟仿真云平台在教育应用中的优势和意义

虚拟仿真云实验教学平台作为一种新型的教学方法&#xff0c;近年来在高校教育中得到了十分广泛的应用。它通过模拟真实的实验场景和实验操作&#xff0c;让学生在计算机上进行实验操作和数据处理&#xff0c;为学生提供了更加便捷、可靠、有效的实验学习环境。本文&#xff0c;…...

CPU的的处理流程如何快速记忆

为了快速记忆CPU的处理流程&#xff0c;可以将其简化成五个主要阶段&#xff0c;通常称为“冯诺依曼架构”的五个基本步骤&#xff0c;或者是流水线处理的几个阶段。下面是一种便于记忆的简化版本&#xff1a; CPU处理流程的五个阶段&#xff1a; 取指令&#xff08;Instructi…...

AI视频教程下载:基于OpenAl、LangChain、 Replicate开发AI应用程序

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

【C++】继承相关(基类与派生类的继承关系以及细节整理)

目录 00.引言 01.继承的定义 02.基类和派生类对象 03.继承中的作用域 04.派生类的默认成员函数 05.友元、静态成员 00.引言 继承是面向对象编程中的一个重要概念&#xff0c;它的作用是创建一个新的类&#xff0c;该类可以从一个已存在的类&#xff08;父类/基类&#x…...

【Web后端】监听器Listener

1、简介 用来监听Servlet组件对象状态发生变化的组件可以监听的源包括:ServetRequest、HttpSession、ServletContext当监听到事件源状态发生变化时&#xff0c;会有对应的响应行为 2、使用方法 在web.xml文件中配置 <listener> <listener-class>com.coder.util.…...

C/C++ 初级球球大作战练手

效果演示&#xff1a; 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之字符串的扩展

字符串的扩展 关键的扩展点及其示例&#xff1a; Unicode 表示与处理 JavaScript 共有6种方法可以表示一个字符。codePointAtpos&#xff1a;String.fromCodePoint…codePoints&#xff1a; **字符串的遍历 for … of **字符串方法的增强 includessearchString[, position]&…...

微信小程序开发,构建NPM报错解决

报错信息如下&#xff1a; message&#xff1a; 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这…...

通过钉钉卡片进行工单审批

我们通常通过钉钉机器人来发送通知&#xff0c;提醒审批人名下有待办工单需要处理。这种通知方式仅能提醒审批人到ITSM中处理&#xff0c;审批人需要打开电脑登陆平台处理&#xff0c;我们就考虑是否能有一种方式能够满足移动端审批&#xff1f; 这里我们可以使用ITSM的移动端版…...

C语言中的控制语句:深入解析与案例实践2

五、使用goto实现无条件跳转 虽然goto语句在现代编程实践中往往被避免使用&#xff0c;因为它可能导致代码难以理解和维护&#xff0c;但在某些特定情况下&#xff0c;它仍然可以作为一种解决方案。goto语句允许程序无条件地跳转到代码中指定的标签位置。 案例代码&#xff1a…...

【ARM Cortex-M 系列 2.1 -- Cortex-M7 Debug system registers】

请阅读【嵌入式开发学习必备专栏】 文章目录 Debug system registers中断控制状态寄存器&#xff08;ICSR&#xff09;Debug Halting Control and Status Register, DHCSR Debug 寄存器DCRSR与DCRDRCPU 寄存器读操作CPU 寄存器写操作CPU 寄存器选择CPU 寄存器读写示例 调试故障…...

深入解析C#中的async和await关键字

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

【VUE.js】前端框架——未完成

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

开源的文件压缩和解压缩软件7-Zip

7-Zip是一款开源的文件压缩和解压缩软件&#xff0c;它支持多种压缩格式&#xff0c;包括自有的7z格式以及常见的ZIP、RAR、TAR等格式。下面是对7-Zip的详细介绍&#xff0c;包括其功能特点、使用方法、优缺点等方面的内容。官网&#xff1a;7-Zip 1. 功能特点&#xff1a; 多…...

搜维尔科技:Patchwork 3D 新功能:爆炸视图动画

Patchwork 3D新功能&#xff1a;爆炸视图动画 为什么使用 Patchwork 3D&#xff1f; Patchwork 3D 软件有助于将 CAD 数据转换成真实感的 3D 模型以用于工业用途。 Patchwork 3D 可轻松进行实时渲染&#xff0c;有助于缩短设计周期&#xff0c;或者让您获得效果逼真的渲染图&…...

Redis——RedLock、Zookeeper及数据库实现分布式锁

在分布式系统中&#xff0c;实现分布式锁是确保数据一致性和防止并发问题的重要手段。以下是使用Redis的RedLock算法、ZooKeeper以及数据库实现分布式锁的基本概念和步骤&#xff1a; 1. Redis的RedLock算法 Redis的RedLock算法是Redis官方推荐的一种分布式锁实现方式&#x…...

OpenSSL自签证书并基于Express搭建Web服务进行SSL/TLS协议分析

OpenSSL自签证书并基于Express搭建Web服务进行SSL/TLS协议分析 起因 最近在学习安全协议&#xff0c;大多数实验都是基于Windows下IIS&#xff0c;或者Linux下nginx搭建的Web服务&#xff0c;搭建环境和编写配置文件比较麻烦。而且我有多个不同环境的设备&#xff0c;折腾起来…...

记录一次 vue2 前端项目整合过程

整合成功效果图 具体说明&#xff1a; 项目A是现在的vue2前端项目&#xff0c;项目B是一个开源的工作流前端&#xff0c;项目后端代码已经整合了&#xff0c;就不多提了。这里主要记录下前端整合的过程和思路。 1、开源工作流里面的功能&#xff0c;拷贝到自己对应的vue2项目里…...

物联网五层架构分析

物联网五层架构分析 随着科技的迅速发展&#xff0c;物联网&#xff08;IoT&#xff09;作为日常生活中不可或缺的一部分&#xff0c;已融入人们的生活和工作中。物联网五层架构&#xff0c;包括感知层、网络层、数据层、应用层和业务层&#xff0c;扮演着关键的角色。 感知层 …...

【Java EE】多线程(三)线程状态

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 | 《数据结构与算法》 | 《C生万物》 |《MySQL探索之旅》 |《Web世界探险家》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更…...

FFmpeg常用API与示例(一)—— 工具库篇(av_log、AVDictionary、avio)

工具层 1.av_log 可以设置日志的级别&#xff0c;这个看看名字就明白了&#xff0c;也不用过多的解释。 AV_LOG_PANICAV_LOG_FATALAV_LOG_ERRORAV_LOG_WARNINGAV_LOG_INFOAV_LOG_VERBOSEAV_LOG_DEBUG void test_log() {/ av_register_all();AVFormatContext *pAVFmtCtx NU…...

日志的基本用法

目标 1. 掌握如何设置日志级别 2. 掌握如何设置日志格式 3. 掌握如何将日志信息输出到文件中 1. logging模块 Python中有一个标准库模块logging可以直接记录日志 1.1 基本用法 import logging logging.debug("这是一条调试信息") logging.info("这是一条…...

什么是页分裂、页合并?

数据组织方式 在InnoDB存储引擎中&#xff0c;表数据都是根据主键顺序组织存放的&#xff0c;这种存储方式的表称为索引组织表(index organized table IOT)。 行数据&#xff0c;都是存储在聚集索引的叶子节点上的。而我们之前也讲解过InnoDB的逻辑结构图&#xff1a; 在I…...