C#通过NTP服务器获取NTP时间

C#通过NTP服务器获取NTP时间

注意事项:

  1. 如果NTP服务器地址是域名,如阿里云的NTP服务器地址。需要DNS解析。
  2. NTP使用UDP通讯,默认端口是123
  3. NTP经过很多年的发展,有4个版本号,目前常用的3和4。NTP区分客户端和服务端,客户端角色标志为3。
  4. NTP发送的时间戳是第41到48个字节。获取到的字节需要转为大端序列。
  5. NTP标准协议返回的时间是自1900.1.1 00:00:00开始的毫秒时间数值,需要字节转换为你需要的日期时间。

以下是通过NTP服务器获取NTP时间的代码:

/// <summary>
/// 获取NTP时间
/// </summary>
/// <param name="serverList"></param>
/// <param name="timeoutMilliseconds"></param>
/// <param name="isToLocal"></param>
/// <returns></returns>
/// <exception cref="Exception"></exception>
public static async Task<DateTime> GetNtpTime(List<string> serverList, bool isToLocal = false, int timeoutMilliseconds = 2000)
{List<string> realServerList = new List<string>();bool needUseLastIp = false;if (serverList is null || !serverList.Any()){serverList = NtpServers;needUseLastIp = true;if (!string.IsNullOrEmpty(lastIpAddress)){realServerList.Add(lastIpAddress);if (serverList.Contains(lastIpAddress)){serverList.Remove(lastIpAddress);}}}realServerList.AddRange(serverList);Log.Information($"GetNtpTime realServerList: {JsonConvert.SerializeObject(realServerList)}");foreach (var server in realServerList){try{IPAddress? ipAddress;List<IPAddress> addressList = new List<IPAddress>();if (!IPAddress.TryParse(server, out ipAddress)){// 如果不是IP地址,则尝试DNS解析try{var hostEntry = await Dns.GetHostEntryAsync(server);addressList = hostEntry.AddressList.ToList();}catch (Exception e){Log.Error($"DNS resolution failed: {e.Message} {e.StackTrace}");}}else{addressList.Add(ipAddress);}if (addressList is null || !addressList.Any()){continue;}foreach (var address in addressList){using (var client = new UdpClient()){client.Client.ReceiveTimeout = timeoutMilliseconds;client.Client.SendTimeout = timeoutMilliseconds;// 发送NTP请求var request = new byte[48];request[0] = 0x1B; // 设置版本号4,模式3(客户端)var endPoint = new IPEndPoint(address, 123);client.Send(request, request.Length, endPoint);// 发送请求并接收响应var response = await client.ReceiveAsync();// 验证响应数据长度if (response.Buffer.Length < 48 || !IsValidNtpResponse(response.Buffer))continue;// 解析时间戳(注意:BitConverter默认是小端,需要反转)ulong intPart = BitConverter.ToUInt32(response.Buffer, 40);ulong fractPart = BitConverter.ToUInt32(response.Buffer, 44);// 转换为大端序(NTP使用大端)intPart = SwapEndianness((uint)intPart);fractPart = SwapEndianness((uint)fractPart);// 计算总时间戳ulong ntpTimestamp = (intPart << 32) | (fractPart & 0xFFFFFFFF);double totalSeconds = (double)ntpTimestamp / (double)(1UL << 32);if (needUseLastIp){lastIpAddress = endPoint.Address.ToString();Log.Information($"记录本次获取NTP时间的IP为:{lastIpAddress}");}// 转换为DateTimeDateTime epoch = new DateTime(1900, 1, 1, 0, 0, 0, DateTimeKind.Utc);DateTime ntpTime = epoch.AddSeconds(totalSeconds);if (isToLocal){ntpTime = ntpTime.ToLocalTime();}return ntpTime;}}}catch (Exception ex){Log.Error($"连接服务器 {server} 失败: {ex.Message} {ex.StackTrace}");continue;}}throw new Exception("所有NTP服务器均无法连接!");
}

获取到的时间,如果需要自己进行时区转换,可以通过C#代码从电脑本机获取到标准的时区列表,然后将获取到的NTP时间加上时区的BaseUtcOffset,即可转换到所需要的时区时间。
C#获取时区列表代码如下

var allZones = TimeZoneInfo.GetSystemTimeZones().ToList();

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/80745.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用cmd来创建数据库和数据库表-简洁步骤

创建数据库和表&#xff1a; 1. 按WinR打开“运行”&#xff0c;输入cmd&#xff0c;回车 2. 登录数据库&#xff1a;mysql -u root -p 然后输入密码 3. 创建数据库create database myblog; myblog为数据库名(自定义你的数据库名) &#xff01;注意分号不要漏了&#xff01; …

java工具类

LocalDateTime LocalDateTime可以获取当前时间&#xff1a; LocalDateTime now LocalDateTime.now(); 同时他也可以获取指定时间&#xff1a; LocalDateTime dateTime LocalDateTime.of(2023, 5, 15, 10, 30) 若我们时间值超出了我们的实际情况值&#xff0c;我们将会出现…

02_java的运行机制以及JDKJREJVM基本介绍

1、运行机制 2、JDK&JRE&JVM JDK 基本介绍 &#xff08;1&#xff09; JDK 的全称(Java Development Kit Java开发工具包) JDK JRE java的开发工具 [ java, javac, javadoc, javap等 ] &#xff08;2&#xff09;JDK是提供给Java开发人员使用的&#xff0c;其…

【文心快码】确实有点东西!

这里写自定义目录标题 背景 Electron 开发 Markdown 编辑器全记录提问1&#xff1a;提问2&#xff1a;提问3&#xff1a;提问4&#xff1a;完整项目结构总结 背景 前两天百度在2025 百度AI开发者大会"如何驾驭Coding Agent分会场"上发布了文心快码&#xff0c;注册试…

AI心理健康服务平台项目面试实战

AI心理健康服务平台项目面试实战 第一轮提问&#xff1a; 面试官&#xff1a; 请简要介绍一下AI心理健康服务平台的核心技术架构。在AI领域&#xff0c;心理健康服务的机遇主要体现在哪些方面&#xff1f;如何利用NLP技术提升用户与AI的心理健康对话体验&#xff1f; 马架构…

Win10安装 P104-100 驱动

安装完之后总结一下, 之前做了不少功课, 在网上搜了很多教程, 视频的文字的, 但是很多已经比较陈旧了. 最后发现的这个 GitHub 项目 NVIDIA-patcher 是最有用的, 因为这是现在这些魔改驱动的来源. NVIDIA-patcher 仓库地址: https://github.com/dartraiden/NVIDIA-patcher 安…

把一个 PyTorch 的图像张量转换成 NumPy 格式,并按照正确的维度顺序显示出来

示例代码&#xff1a; plt.imshow(np.transpose(tensor_denorm.numpy(), (1, 2, 0)))它的作用是&#xff1a;把一个 PyTorch 的图像张量转换成 NumPy 格式&#xff0c;并按照正确的维度顺序显示出来。 &#x1f680; 一步步解释&#xff1a; ✅ tensor_denorm 这是一个形状为…

【速写】conda安装(linux)

序言 昨天叶凯浩空降&#xff08;全马241&#xff09;&#xff0c;降维打击&#xff0c;10分24秒断层夺冠。 夏潇阳10分53秒绝杀小崔10分54秒第2&#xff0c;小崔第3&#xff0c;均配都在3’30"以内&#xff0c;即便我是去年巅峰期也很难跑出这种水平。我就知道他去年大…

算法题(135):唯一的雪花

审题&#xff1a; 本题需要我们对于每一组数据都找出最大的包裹大小 思路&#xff1a; 本题解析题目意思后我们可以把雪花的编号当成数组中元素的值&#xff0c;把包裹看成一个区间。 本质上就是让我们找出一组数据中&#xff0c;所有子段中最长的子段。 方法一&#xff1a;暴力…

算法习题-力扣446周赛题解

算法可以调度思维&#xff0c;让程序员的思维发散&#xff0c;找到更好的解决方案。 第一题&#xff1a;执行指令后的得分 题目&#xff1a; 给你两个数组&#xff1a;instructions 和 values&#xff0c;数组的长度均为 n。你需要根据以下规则模拟一个过程&#xff1a; 从下标…

Ubuntu下MySQL的安装

Ubuntu下MySQL的安装 1. 查看当前操作系统版本2. 添加MySQL APT源2.1 访问下载页面&#xff0c;并下载发布包2.2 执行安装指令2.3 安装MySQL 3. 查看MySQL状态4. 设置开机自启动 1. 查看当前操作系统版本 通过命令lsb_release -a查看&#xff1a; 2. 添加MySQL APT源 2.1 访问下…

航顺 芯片 开发记录 (一) 2025年4月27日19:23:32

芯片型号: HK32F030MF4P6 第一步:创建工程目录 inc :头文件目录 MDK-ARM : 工程根目录 (新建工程选择该目录) src :相关资源存放位置 官方函数库相关内容 官方函数库大致结构图 ├─HK32F030MLib ├─CMSIS │ ├─CM0 │ │ └─Core │ │ arm_common_table…

Python 数据可视化进阶:精准插入图表到指定 Excel 工作表

Python 数据可视化进阶&#xff1a;精准插入图表到指定 Excel 工作表 在处理数据的过程中&#xff0c;我们常常需要将生成的图表精准地插入到已存在数据的 Excel 文件的指定工作表中。借助 Python 的强大库组合&#xff0c;这一操作得以高效实现。以下是经过优化和注释补充的代…

集成方案 | Docusign + 甄零科技,赋能企业海外业务高效增长!

本文将详细介绍 Docusign 与甄零科技的集成步骤及其效果&#xff0c;并通过实际应用场景来展示 Docusign 的强大集成能力&#xff0c;以证明 Docusign 集成功能的高效性和实用性。 甄零科技是一家专注于数字化合同管理系统的 SaaS 解决方案提供商&#xff0c;致力于为企业打造“…

00-算法打卡-目录

1 数组 01-算法打卡-数组-二分查找-leetcode(704)-第一天-CSDN博客 02-算法打卡-数组-二分查找-leetcode(35)-第二天-CSDN博客 03-算法打卡-数组-二分查找-leetcode(34)-第三天_leetcode 34-CSDN博客 04-算法打卡-数组-二分查找-leetcode(69)-第四天-CSDN博客 05-算法打卡-数组…

剑指Offer(数据结构与算法面试题精讲)C++版——day21

剑指Offer&#xff08;数据结构与算法面试题精讲&#xff09;C版——day21 题目一&#xff1a;数据流的第k大数字题目二&#xff1a;出现频率最高的k个数字题目三&#xff1a;和最小的k个数对附录&#xff1a;源码gitee仓库 题目一&#xff1a;数据流的第k大数字 题目&#xff…

NCCL非阻塞non-blocking实现

NCCL (NVIDIA Collective Communications Library) 主要设计用于高性能的集体通信&#xff08;如all-reduce、broadcast等&#xff09;&#xff0c;但其核心函数默认是阻塞式的&#xff08;blocking&#xff09;&#xff0c;即函数返回时操作已完成。不过&#xff0c;你可以通过…

代码随想录算法训练营第60期第二十天打卡

大家好&#xff0c;今天我们继续进入二叉树的章节&#xff0c;二叉树章节应该已经过半了&#xff0c;大家再坚持一下&#xff0c;那么废话不多说&#xff0c;我们继续今天的内容。 第一题对应力扣编号为235的二叉搜索树的最近公共祖先 其实我们上次任务就接触过了二叉树的最近…

8.0 西门子PLC的S7通讯解析

PC与西门子PLC的S7通讯主要有如下几个步骤: 1. TCP的三次握手(由Socket对象自动完成) 2.发送访问请求:COTP 3. 交换通讯信息:setup Commnunication 一、发送访问请求:COTP 比如向PLC请求+以及PLC返回响应的一个实际例子如下: 发送PLC:----> 03 00 00 16 11 E0 …

Nacos-SpringBoot 配置无法自动刷新问题排查

背景 Nacos SpringBoot版本中&#xff0c;提供了NacosValue注解&#xff0c;支持控制台修改值时&#xff0c;自动刷新&#xff0c;但是今天遇见了无法自动刷新的问题。 环境 SpringBoot 2.2.x nacos-client&#xff1a;2.1.0 nacos-config-spring-boot-starter&#xff1a;0…