一個緩存引發(fā)的時區(qū)問題

事件現(xiàn)象

在進行 公司 asp.net core 項目開發(fā)的時候, 出現(xiàn)一個有趣的bug, 有的接口返回的 datetime 類型為后綴帶Z eg `2021-11-24T12:31:50.2246113Z` 的 UTC時間, 有的接口返回的是 正常的 YYYY:MM:DD HH:mm:ss eg:2021-11-24 20:31:56. 這種現(xiàn)象導致前端進行適配的時候 非常困難.
但是因為業(yè)務中對時分秒的要求沒有太高, 所以這個現(xiàn)象一直沒有引起太大的注意, 直到最近開始注意到這個現(xiàn)象

調(diào)查過程

- 一開始很自然的以為是 Newtonsoft.json  或者 System.Text.json 序列化的問題. 因為是公司封裝了一個 Controller base類型. 替換了 asp.net core ActionResult. 但是經(jīng)過幾番排查和試錯發(fā)現(xiàn)這種現(xiàn)象還是存在. 事情進入死胡同
- 然后就發(fā)現(xiàn)經(jīng)過ef core 加載出來的數(shù)據(jù)的 Datetime 類型的 DatetimeKind 是Utc. 這里應該是local 才對. 然后就懷疑是不是 ef core 和 MSSQL 交互的時候出現(xiàn)了什么問題.
- 但是翻了翻 efcore 的源碼 發(fā)現(xiàn)沒有什么地方 明確改變了Datetime 的Kind
- 于是只能去看這個接口的實現(xiàn)代碼, 發(fā)現(xiàn)有一層緩存. 那么很自然的懷疑到這一層緩存上面.
- 公司緩存數(shù)據(jù) 用的是 MessagePack 做格式化, 和 redis做存儲. 
- 那么嫌疑人只能是 MessagePack 和 redis了 

測試代碼

    [Route("type")]
    [ApiController]
    public class MyDatetimeController : AmiController
    {
        private readonly ICache cache;

        public MyDatetimeController(ICache cache)
        {
            this.cache = cache;
        }
        [Route("no-cache")]
        public async Task<AmiResult> Get()
        {
            return Ok(new MyDatetime { Mydate = DateTime.Now, MydateUtc = DateTime.UtcNow });
        }
        [Route("cache")]
        public async Task<AmiResult> GetCache()
        {
            var m = new MyDatetime { Mydate = DateTime.Now, MydateUtc = DateTime.UtcNow };
            cache.Cache(CacheKey.DV, "testd", m);
            cache.TryGetValue(CacheKey.DV, "testd", out m);
            return Ok(m);
        }
    }
    [MessagePack.MessagePackObject]
    public class MyDatetime
    {
        [MessagePack.Key(1)]
        public DateTime MydateUtc { get; set; }
        [MessagePack.Key(2)]
        public DateTime Mydate { get; set; }
    }

測試結果


http://localhost:8011/amiapicore/api/type/no-cache  
{
  "content": {
    "mydateUtc": "2021-11-24T12:31:56.1622685Z",
    "mydate": "2021-11-24T20:31:56.1622681+08:00"
  },
  "errorCode": 0
}

http://localhost:8011/amiapicore/api/type/cache
{
  "content": {
    "mydateUtc": "2021-11-24T12:31:50.2246113Z",
    "mydate": "2021-11-24T12:31:50.2245647Z"
  },
  "errorCode": 0
}

結論

- 很自然的看出, 是MessagePack 在序列化的時候有點不一樣,事實也確實如此 找到了 MessagePack的官方文檔, 上面寫著
DateTime is serialized to MessagePack Timestamp format, it serialize/deserialize UTC and loses Kind info and requires that MessagePackWriter.OldSpec == false. If you use the NativeDateTimeResolver, DateTime values will be serialized using .NET's native Int64 representation, which preserves Kind info but may not be interoperable with non-.NET platforms.
- 簡單翻譯一下就是 DatetTime 被序列化成了 MessagePack Timestamp 格式, 并去掉了 Datetime 的 Kind 信息 以便和其他的系統(tǒng)交流.
- 如果你想保留Kind 信息,只能使用 NativeDateTimeResolver, 但是這樣就只能與 .Net 平臺交互.
### 解決方案

```c#
    [MessagePack.MessagePackObject]
    public class MyDatetime
    {
        [MessagePack.Key(1)]
        public DateTime MydateUtc { get; set; }
        // 簡單的話就加上下面的Atrribute, 讓MessagePack 先不做格式轉(zhuǎn)換, 復雜的話就要和公司的業(yè)務結合起來考慮了;
        [MessagePack.Key(2)]
        [MessagePack.MessagePackFormatter(typeof(MessagePack.Formatters.NativeDateTimeFormatter))]
        public DateTime Mydate { get; set; }
    }

引用

MessagePack-Csharp

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市鹉动,隨后出現(xiàn)的幾起案子轧坎,更是在濱河造成了極大的恐慌,老刑警劉巖泽示,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件缸血,死亡現(xiàn)場離奇詭異,居然都是意外死亡械筛,警方通過查閱死者的電腦和手機捎泻,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來埋哟,“玉大人笆豁,你說我怎么就攤上這事。” “怎么了闯狱?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵煞赢,是天一觀的道長。 經(jīng)常有香客問我哄孤,道長照筑,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任瘦陈,我火速辦了婚禮凝危,結果婚禮上,老公的妹妹穿的比我還像新娘双饥。我一直安慰自己媒抠,他們只是感情好,可當我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布咏花。 她就那樣靜靜地躺著趴生,像睡著了一般。 火紅的嫁衣襯著肌膚如雪昏翰。 梳的紋絲不亂的頭發(fā)上苍匆,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天,我揣著相機與錄音棚菊,去河邊找鬼浸踩。 笑死,一個胖子當著我的面吹牛统求,可吹牛的內(nèi)容都是我干的检碗。 我是一名探鬼主播,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼码邻,長吁一口氣:“原來是場噩夢啊……” “哼折剃!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起像屋,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤怕犁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后己莺,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體奏甫,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年凌受,在試婚紗的時候發(fā)現(xiàn)自己被綠了阵子。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡胜蛉,死狀恐怖款筑,靈堂內(nèi)的尸體忽然破棺而出智蝠,到底是詐尸還是另有隱情腾么,我是刑警寧澤奈梳,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站解虱,受9級特大地震影響攘须,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜殴泰,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一于宙、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧悍汛,春花似錦捞魁、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽查描。三九已至,卻和暖如春昆著,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背术陶。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工凑懂, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人梧宫。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓接谨,卻偏偏與公主長得像,于是被迫代替她去往敵國和親塘匣。 傳聞我的和親對象是個殘疾皇子脓豪,可洞房花燭夜當晚...
    茶點故事閱讀 44,864評論 2 354

推薦閱讀更多精彩內(nèi)容