如何从NTP服务器获取准确的时间?
创作时间:
作者:
@小白创作中心
如何从NTP服务器获取准确的时间?
引用
1
来源
1.
https://shuyeidc.com/wp/11189.html
网络时间协议(NTP)是一种用于使计算机时钟同步到互联网标准时间的协议。通过NTP,我们可以让系统的时钟保持与标准时间的一致性,这对于需要高精度时间同步的场景非常重要,如日志记录和定时任务等。本文将详细介绍如何从NTP服务器获取准确的时间,包括NTP协议的基本概念、包结构、UNIX时间戳与NTP时间戳的区别,并提供具体的代码实现步骤。
一、简介
网络时间协议(NTP)是一种用于使计算机时钟同步到互联网标准时间的协议,通过NTP,我们可以让系统的时钟保持与标准时间的一致性,这对于需要高精度时间同步的场景非常重要,如日志记录和定时任务等。
二、NTP包结构
NTP包由48字节组成,包含以下字段:
- LI(Leap Indicator):2位,表示闰秒信息。
- VN(Version Number):3位,表示NTP版本号。
- Mode:3位,表示NTP的工作模式。
- Stratum:8位,表示服务器层级。
- Poll:8位,表示连续NTP请求间的最大间隔。
- Precision:8位,表示服务器时钟的精度。
- Root Delay:32位,表示从NTP客户端到NTP服务器的根延迟。
- Root Dispersion:32位,表示NTP服务器与其上一级时钟源的偏差。
- Reference Identifier:32位,表示NTP服务器的参考标识符。
- Reference Timestamp:64位,表示参考时间戳。
- Originate Timestamp:64位,表示发起时间戳。
- Receive Timestamp:64位,表示接收时间戳。
- Transmit Timestamp:64位,表示传输时间戳。
三、UNIX时间戳和NTP时间戳
UNIX时间戳是从1970年1月1日00:00:00 UTC开始计算的秒数,而NTP时间戳则是从1900年1月1日00:00:00 UTC开始计算的秒数,两者之间的偏移量是2208988800秒。
四、代码实现
在Linux环境下,使用标准的POSIX/BSD套接字编程来获取NTP时间,以下是具体步骤:
- NTP服务端地址
这里以阿里云的NTP服务器ntp3.aliyun.com为例,端口为123。
#define NTP_SERVER "ntp3.aliyun.com" // NTP服务器地址
#define NTP_PORT 123 // 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 originate_timestamp[2]; // 发起时间戳
uint32_t receive_timestamp[2]; // 接收时间戳
uint32_t transmit_timestamp[2]; // 传输时间戳
} ntp_packet;
- 创建套接字并设置超时时间
int sockfd;
struct timeval timeout;
timeout.tv_sec = NTP_RCV_TIMEO / 1000;
timeout.tv_usec = (NTP_RCV_TIMEO % 1000) * 1000;
sockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout));
- 设置服务端地址和NTP请求报文
struct sockaddr_in server_addr;
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(NTP_PORT);
inet_pton(AF_INET, NTP_SERVER, &server_addr.sin_addr);
- 发送请求并接收响应
ntp_packet packet;
memset(&packet, 0, sizeof(ntp_packet));
packet.li_vn_mode = 0x1B; // 设定模式为客户模式
sendto(sockfd, packet, sizeof(ntp_packet), 0, (struct sockaddr *)&server_addr, sizeof(server_addr));
recvfrom(sockfd, packet, sizeof(ntp_packet), 0, NULL, NULL);
- 计算时间偏移和调整系统时间
double offset = (double)(((int32_t)packet.transmit_timestamp[0] (int32_t)packet.originate_timestamp[0]) + (double)packet.transmit_timestamp[1] / 4294967296 (double)packet.originate_timestamp[1] / 4294967296);
offset *= -1; // 因为NTP时间戳起点比UNIX时间戳早70年,所以需要取反
time_t now = time(NULL) + offset;
struct tm *tm_info = localtime(&now);
printf("当前时间: %s", asctime(tm_info));
通过以上步骤,我们可以利用NTP协议获取准确的网络时间,并在本地系统中进行调整,需要注意的是,NTP协议涉及复杂的时间计算和网络通信过程,因此在实际应用中应确保每一步操作的正确性。
热门推荐
如何简化生育津贴申请流程?
“奇点”的“奇”正确读法是什么?宇宙年龄竟不是138亿年?
勾股定理还能这样证明?高中生一连发现10种证明方法,陶哲轩点赞
《沁园春·长沙》赏析:谁主沉浮?
离职出国人员退休金如何领取?
出国后养老保险怎么领取?一文详解境外养老保障政策
浅析作业成本法在企业成本管理中的应用
洋葱热量(减肥蔬菜排行榜前十名)
信用卡刷卡手续费:商家支付比例及行业差异全解析
云南芒市的“树包塔”:自然与人文完美交融的地标
以诗迎春,勾勒济南美
如何判断市场趋势?市场趋势分析的方法有哪些?
日本旅游签证办理指南:提前1-2个月准备,畅游东瀛不是梦
技嘉Z790主板在Win11系统下如何优化?兼容性如何?
酒的三大类型?正确的酒类的三大分类
谷曙光教授在孔学堂讲述唐宋八大家的八个历史现场
22所高校新晋A+!2024中国最好学科排名
新手入门:合约交易怎么玩?掌握基础知识,做好风险控制
国产ChirpIoT™扩频技术原理以及模块应用详解
中国劳工奋斗史:双休与 8 小时工作制的来之不易
1000个能用的实名认证:全面解析与合法使用指南
李商隐最意难平的一首诗,短短八句全是旷世名句,最后两句传为爱情千古绝唱
术后营养关键饮食怎么吃?不建议...?
《大小姐她人设崩了》:一部关于成长与蜕变的短剧佳作
工伤伤残申请表需要什么材料和手续
药浴包配方与功效:如何根据体质选择最适合的药浴?
云计算与传统数据中心的区别
减肥到底能不能吃麻辣烫?
秋天口鼻干、咳嗽、皮肤干燥?这几种养阴润燥的中药安排上
如何在不删除C/D盘文件的基础上把C盘多余的空间分给D盘