.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

WeskyNet發表於2024-05-15

對某個遠端伺服器啟用和設定NTP服務(Windows系統)

開啟登錄檔

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

將 Enabled 的值設定為 1,這將啟用NTP伺服器功能。

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

防火牆開放UDP 123埠

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

開啟“服務”應用(可以在開始選單搜尋“服務”),找到“Windows Time”服務。右鍵點選“Windows Time”服務,選擇“重啟”。

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

執行以下命令來配置伺服器模式並重啟時間服務:

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

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

客戶端上面,輸入以下命令,替換你自己的NTP服務端IP即可,如果顯示類似以下的時間輸出,說明是正常的。

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

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

編寫程式,進行時間同步

引用包Wesky.Net.OpenTools ,版本選擇1.0.6或以上版本。

該包的自述檔案內容供參考:

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

在程式裡面使用,以下測試內容供參考。其中,ntpServer可以是ip地址也可以是ntp伺服器的域名地址或者網際網路ntp伺服器地址等。獲取時間預設埠號沒指定的話是123,如果要指定,只需要在引數裡面新增埠號引數即可。

.NET實現獲取NTP伺服器時間並同步(附帶Windows系統啟用NTP服務功能)

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伺服器時間核心程式碼解析如下:

 1 /// <summary>
 2 /// 獲取NTP伺服器的時間。
 3 /// Retrieves the time from an NTP server.
 4 /// </summary>
 5 /// <param name="ntpServer">NTP伺服器地址 | NTP server address</param>'
 6 /// <param name="ntpPort">NTP服務的埠 | NTP service port</param>
 7 /// <returns>伺服器時間 | Server time</returns>
 8 public static DateTime GetNtpServerTime(string ntpServer,int ntpPort=123)
 9 {
10     // 初始化NTP資料緩衝區
11     // Initialize NTP data buffer
12     byte[] ntpData = new byte[NtpDataLength];
13     ntpData[0] = 0x1B; // NTP version number (3) and mode (3), client request
14 
15     var addresses = Dns.GetHostAddresses(ntpServer);
16     IPEndPoint ipEndPoint = new IPEndPoint(addresses[0], ntpPort);
17 
18     using (var socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp))
19     {
20         socket.Connect(ipEndPoint);
21         socket.Send(ntpData);
22         socket.Receive(ntpData);
23     }
24 
25     // 從位元組40和44提取時間戳
26     // Extract timestamp from bytes 40 and 44
27     ulong intPart = BitConverter.ToUInt32(ntpData, 40);
28     ulong fractPart = BitConverter.ToUInt32(ntpData, 44);
29 
30     // 轉換位元組序為小端格式
31     // Convert byte order to little endian
32     intPart = SwapEndianness(intPart);
33     fractPart = SwapEndianness(fractPart);
34 
35     var milliseconds = (intPart * 1000) + ((fractPart * 1000) / 0x100000000L);
36 
37     // NTP時間是從1900年開始計算的,這裡將其轉換為UTC時間
38     // NTP time starts from 1900, this converts it to UTC DateTime
39     DateTime networkDateTime = new DateTime(1900, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddMilliseconds((long)milliseconds);
40 
41     return networkDateTime.ToLocalTime();
42 }

相關文章