您好,登錄后才能下訂單哦!
很多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
免責聲明:本站發(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)容。