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

C#中的DateTime在邏輯上有個很是嚴重的缺陷:html

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

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

緣由在於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 。post

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

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

複製代碼
> 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時間的怪物。code

當d3轉成本地時間時,會把d3做爲UTC時間來加8小時。htm

當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時間戳,都能勝任。空間代價也很是小,除非你的存儲空間真的是硬傷。。

 

 

出處:http://www.cnblogs.com/zlmdy/p/8560396.html

相關文章
相關標籤/搜索