根據Time Protocol從NIST Internet Time Servers獲取準確時間

来源:http://www.cnblogs.com/zhangxiaobin/archive/2017/09/08/7493469.html
-Advertisement-
Play Games

Time Protocol(RFC-868)是一種非常簡單的應用層協議:它返回一個32位的二進位數字,這個數字描述了從1900年1月1日0時0分0秒到現在的秒數,伺服器在TCP的37號埠監聽時間協議請求。本函數將伺服器返回值轉化成本地時間。 先前不知道有現成的IPAddress.NetworkTo ...


Time Protocol(RFC-868)是一種非常簡單的應用層協議:它返回一個32位的二進位數字,這個數字描述了從1900年1月1日0時0分0秒到現在的秒數,伺服器在TCP的37號埠監聽時間協議請求。本函數將伺服器返回值轉化成本地時間。

先前不知道有現成的IPAddress.NetworkToHostOrder函數,所以自己直接寫了個ReverseBytes函數,把位元組數組從Big-endian轉換為Little-endian。這個函數可能在其他地方也有用,所以索性就留著了。

 

 1         private const int BUFSIZE = 4;      //字元數組的大小
 2         private const int PORT = 37;        //伺服器埠號
 3         private const int TIMEOUT = 3000;   //超時時間(毫秒)
 4         private const int MAXTRIES = 3;     //嘗試接受數據的次數
 5 
 6         /// <summary>
 7         /// 從NIST Internet Time Servers獲取準確時間。
 8         /// </summary>
 9         /// <param name="dateTime">返回準確的本地時間</param>
10         /// <param name="timeServer">伺服器列表</param>
11         /// <returns>獲取時間失敗將返回false,否則返回true</returns>
12         public static bool GetDateTimeFromTimeServer(out DateTime now, string timeServers = "time.nist.gov")
13         {
14             Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
15             //設置獲取超時時間
16             socket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReceiveTimeout, TIMEOUT);
17 
18             byte[] rcvBytes = new byte[BUFSIZE];    //接收數據的位元組數組
19             int tries = 0;                          //記錄嘗試次數
20             bool received = false;                  //接收是否成功
21             int totalBytesRcvd = 0;                 //總共接收的位元組數
22             int bytesRcvd = 0;                      //本次接收的位元組數
23             do
24             {
25                 try
26                 {
27                     socket.Connect(Dns.GetHostEntry(timeServers).AddressList, PORT);
28                     while ((bytesRcvd = socket.Receive(rcvBytes, totalBytesRcvd, BUFSIZE - totalBytesRcvd, SocketFlags.None)) > 0)
29                     {
30                         totalBytesRcvd += bytesRcvd;
31                     }
32                     received = true;
33                 }
34                 catch (SocketException)
35                 {
36                     //超時或者其他Socket錯誤,增加參數次數
37                     tries++;
38                 }
39             } while ((!received) && (tries < MAXTRIES));
40             socket.Close();
41 
42             if (received)
43             {
44                 //將位元組數組從Big-endian轉換為Little-endian
45                 //ReverseBytes(ref rcvBytes, 0, 4);
46                 //UInt32 seconds = BitConverter.ToUInt32(rcvBytes, 0);
47                 UInt32 seconds = BitConverter.ToUInt32(rcvBytes, 0);
48                 if (BitConverter.IsLittleEndian)
49                 {
50                     seconds = (UInt32)IPAddress.NetworkToHostOrder((int)seconds);
51                 }
52                 //從1900年1月1日0時0分0秒日期加上獲取的秒數並轉換到當前本地時區時間
53                 now = new DateTime(1900, 1, 1, 0, 0, 0, DateTimeKind.Utc).AddSeconds(seconds).ToLocalTime();
54                 return true;
55             }
56             else
57             {
58                 now = DateTime.Now;
59                 return false;
60             }
61         }
62 
63         /// <summary>
64         /// 翻轉byte數組的位元組順序
65         /// </summary>
66         /// <param name="bytes">要翻轉的位元組數組</param>
67         /// <param name="start">規定轉換起始位置</param>
68         /// <param name="len">要翻轉的長度</param>
69         private static void ReverseBytes(ref byte[] bytes, int start, int len)
70         {
71             if ((start < 0) || (start > bytes.Length - 1) || (len > bytes.Length))
72             {
73                 throw new ArgumentOutOfRangeException();
74             }
75 
76             int end = start + len - 1;
77             if (end > bytes.Length)
78             {
79                 throw new ArgumentOutOfRangeException();
80             }
81 
82             byte tmp;
83             for (int i = 0, index = start; index < start + len / 2; index++, i++)
84             {
85                 tmp = bytes[end - i];
86                 bytes[end - i] = bytes[index];
87                 bytes[index] = tmp;
88             }
89         }

代碼未經過嚴格測試,如果有什麼錯誤,歡迎指出,謝謝!

 

參考文獻

[1]陳香凝,王燁陽,陳婷婷,張錚.Windows網路與通信程式設計第三版[M].人民郵電出版社,2017:27-28.

[2]D.Makofske,M.Donahoo,K.Calvert.TCPIP Sockets in C# Practical Guide for Programmers[M].Morgan Kaufmann.2004。

[3]NIST Internet Time Servers.http://tf.nist.gov/tf-cgi/servers.cgi.


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 問題現象:由於重裝linux,並且加了固態硬碟,直接將系統裝在固態硬碟中。啟動伺服器的時候, 便看不到原來機械硬碟的掛載目錄了,不知如何訪問機械硬碟了。直接用命令 mount /dev/sda3 /store 掛載, 提示 mount: unknown filesystem type 'LVM2_m ...
  • Linux對於記憶體的管理涉及到非常多的方面,這篇文章首先從對進程虛擬地址空間的管理說起。(所依據的代碼是2.6.32.60) 無論是內核線程還是用戶進程,對於內核來說,無非都是task_struct這個數據結構的一個實例而已,task_struct被稱為進程描述符(process descripto ...
  • 首先找出所有可選的佈局(layout)方案: 可以看到 us 下有很多常見的佈局方案(以下為節選): 但文檔可能不全,比如早在 13 年加入的 norman 方案就未在 man 手冊列出。 可以通過搜索 symbol 文件找到: 更改佈局方案: 加上 / 選項會顯示輸出信息: 切換回主流的 QWER ...
  • 轉:http://blog.csdn.net/hongchangfirst/article/details/7075026 大家都知道進程,可是知道linux是怎麼管理其進程的嗎?每一個進程都有一個進程描述符,具體是task_struct結構體存儲相關的信息,在linux/sched.h文件里定義, ...
  • 一、分佈位置上的區別: kmalloc()和__get_free_pages()函數申請的記憶體位於物理記憶體的映射區域,而且在物理上也是連續的,它們與真實的物理地址只有一個固定的偏移,因此存在簡單的線性關係;(3G+896M)(低端記憶體); vmalloc函數申請的虛擬記憶體與物理記憶體之間也沒有簡單的換 ...
  • 添加樣式: 在html中,需要創建2層div來實現。一個div包含另一個div: 效果: ...
  • 表格控制項 Spread Studio 發佈了全新的 V11 CTP 版本。在此版本中,Spread For WinForms 引入了 Spread Common,也帶來了 Spread 性能的巨大提升和記憶體消耗的急劇下降。 ...
  • 1. 前言 做了WPF開發多年,一直未曾自己實現一個自定義Window Style,無論是《WPF編程寶典》或是各種博客都建議使用WindowStyle="None" 和 AllowsTransparency="True",於是想當然以為這樣就可以了。最近來了興緻想自己實現一個,才知道WindowS ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...