.NET实现获取NTP服务器时间并同步(附带Windows系统启用NTP服务功能)

对某个远程服务器启用和设置NTP服务(Windows系统)

打开注册表

HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer

将 Enabled 的值设置为 1,这将启用NTP服务器功能。

a57dc76a623a35d23e84f9807268b4e3.png

防火墙开放UDP 123端口

a9efd8a2cac792b63ec56ef48b44d082.png

打开“服务”应用(可以在开始菜单搜索“服务”),找到“Windows Time”服务。右键点击“Windows Time”服务,选择“重启”。

3bfdb1f5d8671e1b24bdedb289edec26.png

执行以下命令来配置服务器模式并重启时间服务:

w32tm /config /reliable:YES /update
net stop w32time
net start w32time

6af8ad27a2f4734f0a5096d2dc6a388d.png

客户端上面,输入以下命令,替换你自己的NTP服务端IP即可,如果显示类似以下的时间输出,说明是正常的。

w32tm /stripchart /computer:ip地址 /samples:5 /dataonly

3f0c867d2409a6a4f528a9e3047c6380.png

 编写程序,进行时间同步

引用包Wesky.Net.OpenTools ,版本选择1.0.6或以上版本。

a1484f4bd508d9eb3d0a10d9294ea8a7.png

该包的自述文件内容供参考:

https://www.nuget.org/packages/Wesky.Net.OpenTools/1.0.6#readme-body-tab

029aa664a022c696fcede59336d5d17a.png

 在程序里面使用,以下测试内容供参考。其中,ntpServer可以是ip地址也可以是ntp服务器的域名地址或者互联网ntp服务器地址等。获取时间默认端口号没指定的话是123,如果要指定,只需要在参数里面新增端口号参数即可。

1cf8737153be92fe577403c91f913672.png

static void Main(string[] args)
{
    string ntpServer = "ip";
    Console.WriteLine($"当前时间:\r\n{ DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss ms")}");
    DateTime time = NtpClient.GetNtpServerTime(ntpServer);
    Console.WriteLine($"获取到的时间为:\r\n {time.ToString("yyyy/MM/dd HH:mm:ss ms")}");
    NtpClient.SetSystemTime(time);
    Console.WriteLine($"更改后的系统时间:\r\n{ DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss ms")}");
    Console.ReadLine();
}

获取ntp服务器时间核心代码解析如下:

/// <summary>
/// 获取NTP服务器的时间。
/// Retrieves the time from an NTP server.
/// </summary>
/// <param name="ntpServer">NTP服务器地址 | NTP server address</param>'
/// <param name="ntpPort">NTP服务的端口 | NTP service port</param>
/// <returns>服务器时间 | Server time</returns>
public static DateTime GetNtpServerTime(string ntpServer,int ntpPort=123)
{
    // 初始化NTP数据缓冲区
    // Initialize NTP data buffer
    byte[] ntpData = new byte[NtpDataLength];
    ntpData[0] = 0x1B; // NTP version number (3) and mode (3), client request


    var addresses = Dns.GetHostAddresses(ntpServer);
    IPEndPoint ipEndPoint = new IPEndPoint(addresses[0], ntpPort);


    using (var socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp))
    {
        socket.Connect(ipEndPoint);
        socket.Send(ntpData);
        socket.Receive(ntpData);
    }


    // 从字节40和44提取时间戳
    // Extract timestamp from bytes 40 and 44
    ulong intPart = BitConverter.ToUInt32(ntpData, 40);
    ulong fractPart = BitConverter.ToUInt32(ntpData, 44);


    // 转换字节序为小端格式
    // Convert byte order to little endian
    intPart = SwapEndianness(intPart);
    fractPart = SwapEndianness(fractPart);


    var milliseconds = (intPart * 1000) + ((fractPart * 1000) / 0x100000000L);


    // NTP时间是从1900年开始计算的,这里将其转换为UTC时间
    // NTP time starts from 1900, this converts it to UTC DateTime
    DateTime networkDateTime = new DateTime(1900, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddMilliseconds((long)milliseconds);


    return networkDateTime.ToLocalTime();
}

如果觉得有帮助,欢迎点赞、转发和在看。谢谢大家。

也欢迎关注个人公众号:

89fdd54e0822f305a599531eed9f31b7.jpeg

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值