溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Json.NET特殊處理64位長整型數(shù)據(jù)

發(fā)布時間:2020-08-05 14:53:15 來源:網(wǎng)絡 閱讀:3110 作者:邊城__ 欄目:編程語言

  很多ASP.NET項目,尤其是使用了Ajax的項目,常常需要返回JSON格式的數(shù)據(jù)。.NET框架從3.5版本開始提供了JSON的序列化和反序列化工具,不過個人感覺不太好用,后來找了第三方的Newtonsoft.Json來用。再后來,在MVC4中,微軟已經(jīng)默認使用Json.NET(Newtonsoft.Json)來處理JSON數(shù)據(jù)了。


  JavaScript數(shù)值精度是32位,如果整數(shù)數(shù)度超過32位,就會被當作浮點數(shù)處理。換句話說,如果從服務端生成的JSON,某個值是64位整數(shù),傳到前端JavaScript,再傳回服務端,不做任何運算,都可能出現(xiàn)失真。做個實驗:


> var a = 123456789012345678
> console.log(a);
123456789012345680


  很要命的一點是,數(shù)據(jù)庫設(shè)計中常常會用bigint(64位)整數(shù)來作為主鍵,是一個非常重要而且不能有偏差的數(shù)據(jù),比如,一個模型:


// C# 匿名對象
new {
    id: 123456789012345678L,
    name: "James"
};


  轉(zhuǎn)換成JSON輸出到前端是:


{"id":123456789012345678,"name":"James"}


  通過Ajax取得的對象輸出就有點不妙了


$.getJSON("/api/test").done(function(jo) {
    console.log(jo);
});
// Object {id: 123456789012345680, name: "James"}


  顯然,這個對象修改數(shù)值之后再傳回服務器,就會找不到主鍵,或者更新成錯誤的數(shù)據(jù),造成一個不易發(fā)現(xiàn)的巨大BUG。


  解決辦法當然是有的,JavaScript處理字符串的能力非常強,完全可以把服務器端的64位整數(shù)處理成字符串類型。不過 Json.NET 默認是把 long 處理成 number 類型的,如果要處理成 string 類型,需要自定義一個JsonConverter。


  考慮到用十六進制表示的整數(shù)看起來比較整齊,所以定義一個HexLongConverter來轉(zhuǎn)換long/ulong型數(shù)據(jù)與16進制表示的字符串。


public class HexLongConverter : JsonConverter
{
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        // 由于CanConvert過濾,數(shù)據(jù)類型只可能是long或ulong
        // 統(tǒng)一轉(zhuǎn)換成long類型處理
        long v = value is ulong ? (long)(ulong)value : (long)value;
        writer.WriteValue(v.ToString("X16"));
    }
    public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
    {
        // 取得讀到的十六進制字符串
        string hex = reader.Value as string;
        // 調(diào)用ToInt64擴展將字符串轉(zhuǎn)換成long型
        // ToInt64擴展方法后附
        long v = hex.ToInt64(NumberStyles.HexNumber, 0L);
        // 將v轉(zhuǎn)換成實際需要的類型 ulong 或 long(不轉(zhuǎn)換)
        return typeof (ulong) == objectType ? (object) (ulong) v : v;
    }
    public override bool CanConvert(Type objectType)
    {
        // 只處理long和ulong兩種類型的數(shù)據(jù)
        switch (objectType.FullName)
        {
            case "System.Int64":
            case "System.UInt64":
                return true;
            default:
                return false;
        }
    }
}


  上面的代碼用到了一個string的擴展方法ToInt32:


public static class StringExtention
{
    public static int ToInt32(this string me, NumberStyles style,
        int defaultValue)
    {
        int? value = me.ToInt32(style);
        return value == null ? defaultValue : value.Value;
    }
}


  在序列化或反序列化模型的時候,只需要加入HexLongConverter對象作為參數(shù)即可:


// 序列化
string json = JsonConvert.SerializeObject(model, new HexLongConverter());
// 反序列化
SomeModal model = JsonConvert.DeserializeObject<model>(json, new HexLongConverter));




相關(guān)鏈接:


[Json.NET]

  • http://json.codeplex.com/

  • http://james.newtonking.com/json



向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI