C#中DateTime的缺陷與代替品DateTimeOffset

来源:https://www.cnblogs.com/zlmdy/archive/2018/03/13/8560396.html
-Advertisement-
Play Games

C#中的DateTime在邏輯上有個非常嚴重的缺陷: 在C#交互模式中輸入以上代碼,可以發現儘管一個是本地時間(d),一個是UTC時間(d2),只是時區不一樣,但在這個世界上,應該屬於同一個時刻。然而兩個時間卻不相等。。。 原因在於DateTime不存儲時區,或者說,只存儲了一個模糊的關於時區的欄位 ...


C#中的DateTime在邏輯上有個非常嚴重的缺陷:

> var d = DateTime.Now;
> var d2 = d.ToUniversalTime();
> d == d2
false
> d.Equals(d2);
false

在C#交互模式中輸入以上代碼,可以發現儘管一個是本地時間(d),一個是UTC時間(d2),只是時區不一樣,但在這個世界上,應該屬於同一個時刻。然而兩個時間卻不相等。。。

原因在於DateTime不存儲時區,或者說,只存儲了一個模糊的關於時區的欄位Kind,它是DateTimeKind枚舉類型的,有三種取值:Utc/Local/Unspecified,當取值為Unspecified時,則會有歧義。

但我還是要吐槽,如果d.Kind或d2.Kind中任意一個是Unspecified,那麼d != d2我可以理解。但是上面的d.Kind是Local,d2.Kind是Utc,如果按照DateTime不存儲時區的邏輯,那麼這兩個統一轉換Utc或者Local時,那麼它們應該相等,事實上也是如此:

> d == d2.ToLocalTime()
true

如果把d的本地時間t1當做9,本地時間所處時區z1當做+8,相應的UTC時間t0當做1,UTC時間所處時區z0當做0,對它們做規範化處理:

那麼 d = t1-z1 = 9 - 8 = 1, d2 = t0 - z0 = 1 - 0 = 1 。

然而 d != d2。這才是它怪異的地方。

以東八區為例,在C#交互模式中輸入以下代碼:

> var d3 = new DateTime(2018, 1, 1);
> d3
[2018/1/1 0:00:00]
> d3.ToLocalTime()
[2018/1/1 8:00:00]
> d3.ToUniversalTime()
[2017/12/31 16:00:00]

可以發現,一個簡單的構造函數,開發者心中預設一般都是本地時間,然而它卻允許直接創建出一個既非本地時間、也非UTC時間的怪物。

當d3轉成本地時間時,會把d3作為UTC時間來加8小時。

當d3轉成UTC時間時,卻會把d3作為本地時間來減8小時。

那麼d3到底是本地時間還是UTC時間呢?沒人清楚,除非它存在於一個非常小的局部作用域中,並且生命周期極短,這時候我們也許可以假設它為本地時間。

然而這個本地時間也依賴於它的運行環境,如果是有幾台時區不一致的電腦,閹割了時區信息的DateTime轉成字元串在網路中傳輸到另一個時區(比如隔壁的十一區)的另一臺伺服器中,解析出來後,所謂的東八區本地時間8點,到了日本,變成了既非本地時間、也非UTC時間的怪物。

DateTime在官方文檔中已經不推薦使用,而是推薦使用它的代替品DateTimeOffset,後者保存時區信息。

在交互模式中驗證一下:

> var dto = DateTimeOffset.Now;
> var dto2 = dto.ToUniversalTime();
> dto == dto2
true

可以發現,DateTimeoffset判斷兩個時間是否等價的標準,是以世界時間軸的時刻來判斷的,與時區無關,甚至可以與UTC時間無關。只要它們都在同一個時間體系裡、能互相變換即可。

在實際項目中,建議大家:

  • 如果有使用DateTime的,統一換成DateTimeOffset。
  • 如果有用到32比特的UNIX時間戳的,統一換成64比特的long來存儲UtcTicks。

即使項目本身不跨時區,仍然有可能遇到時區問題,比如如果使用了mongodb的,mongodb存儲的時候都是統一存成UTC時間的(好像是,忘了),而且一般來說會帶有時區信息。但是有一種情況比較糟糕,如果你的DateTime的Kind是Unspecified的,隱含的時區的信息就會丟失。再取出來之後,就會有8小時的時差。有一些第三方的或者自己公司的類庫之類的,如果沒有處理好這個問題,也有潛在的時區丟失問題。UNIX時間戳存成Utc Ticks也有好處,無論是精度還是時間跨度,都遠超UNIX時間戳。只需要64比特,即可獲得下至100納秒的精度,上超萬年的時間跨度,一勞永逸,無論是轉回UNIX時間戳還是JS時間戳,都能勝任。空間代價也非常小,除非你的存儲空間真的是硬傷。。

 


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

-Advertisement-
Play Games
更多相關文章
  • 一級指針 int *p; //表示定義一個int型(4位元組)的指針p &p //表示p自身的地址位置 p //表示p指向的地址位置(也就是p變數的值) *p //表示p指向的地址裡面的內容 所以 * 的作用: p變數的值作為地址,去訪問這個地址的內容 二級指針 int **pp //表示定義一個in ...
  • Python 元組 元組的定義 元組(tuple)是一種Python對象類型,元組也是一種序列 Python中的元組與列表類似,不同之處元組的元素不能修改 元組使用小括弧,列表使用方括弧 元組的創建,即在括弧中添加元素,並使用逗號隔開 元組是一種序列,序列的基本操作 len() 、+、*、in、ma ...
  • 定時運行任務對於一個網站來說,是一個比較重要的任務,比如定時發佈文檔,定時清理垃圾信息等,現在的網站大多數都是採用PHP動態語言開發的,而對於PHP的實現決定了它沒有Java和.Net這種AppServer的概念,而http協議是一個無狀態的協議,PHP只能被用戶觸發,被調用,調用後會自動退出記憶體, ...
  • 題目描述 火星探險隊的登陸艙將在火星錶面著陸,登陸艙內有多部障礙物探測車。登陸艙著陸後,探測車將離開登陸艙向先期到達的傳送器方向移動。探測車在移動中還必須採集岩石標本。每一塊岩石標本由最先遇到它的探測車完成採集。每塊岩石標本只能被採集一次。岩石標本被採集後,其他探測車可以從原來岩石標本所在處通過。探 ...
  • 首先本人學習變編程有半年了吧,現在是大四準畢業生,今年六月份畢業,專業是機電類的,與IT行業幾乎不相關,所以我是零基礎學習編程的。 看視頻學習 我接觸python編程是看書本自己理解然後自己跟著書本寫代碼,學了一段時間後朋友介紹說上網搜培訓視頻更容易學,於是我去找了一套視頻自己慢慢看慢慢學。接觸視頻 ...
  • 簡單讀取 json 配置文件 背景 目前發現網上的 .NET Core 讀取配置文件有點麻煩,自己想搞個簡單點的。 .NET Core 已經不使用之前的諸如 app.config 和 web.config 等 xml 形式的配置文件,一致採用 json 格式來存儲配置文件信息。 json 文件 de ...
  • 在WPF自學入門(二)WPF-XAML佈局控制項的文章中分別介紹StackPanel,WarpPanel,DockPanel,Grid,Canvas五種佈局容器的使用,可以讓我們大致瞭解容器可以使用在什麼地方。今天我們就來簡單瞭解一下WPF中的三個帶標題的內容控制項,分別是GroupBox,TabCon ...
  • 學了那麼久的函數式編程語言,一直想寫一些相關的文章。經過一段時間的考慮,我決定開這個坑。 至於為什麼選擇C#,在我看來,編程語言分三類:一類是難以進行函數式編程的語言,這類語言包括Java6、C語言等。這類語言由於不支持匿名函數等特性,進行函數式編程會比較困難;一類是自稱“函數式編程語言”的語言,包 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...