很多ASP.NET項(xiàng)目,尤其是使用了Ajax的項(xiàng)目,常常需要返回JSON格式的數(shù)據(jù)。.NET框架從3.5版本開始提供了JSON的序列化和反序列化工具,不過個(gè)人感覺不太好用,后來找了第三方的Newtonsoft.Json來用。再后來,在MVC4中,微軟已經(jīng)默認(rèn)使用Json.NET(Newtonsoft.Json)來處理JSON數(shù)據(jù)了。
成都創(chuàng)新互聯(lián)公司服務(wù)項(xiàng)目包括隆堯網(wǎng)站建設(shè)、隆堯網(wǎng)站制作、隆堯網(wǎng)頁制作以及隆堯網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,隆堯網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到隆堯省份的部分城市,未來相信會繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
JavaScript數(shù)值精度是32位,如果整數(shù)數(shù)度超過32位,就會被當(dāng)作浮點(diǎn)數(shù)處理。換句話說,如果從服務(wù)端生成的JSON,某個(gè)值是64位整數(shù),傳到前端JavaScript,再傳回服務(wù)端,不做任何運(yùn)算,都可能出現(xiàn)失真。做個(gè)實(shí)驗(yàn):
> var a = 123456789012345678 > console.log(a); 123456789012345680
很要命的一點(diǎn)是,數(shù)據(jù)庫設(shè)計(jì)中常常會用bigint(64位)整數(shù)來作為主鍵,是一個(gè)非常重要而且不能有偏差的數(shù)據(jù),比如,一個(gè)模型:
// C# 匿名對象 new { id: 123456789012345678L, name: "James" };
轉(zhuǎn)換成JSON輸出到前端是:
{"id":123456789012345678,"name":"James"}
通過Ajax取得的對象輸出就有點(diǎn)不妙了
$.getJSON("/api/test").done(function(jo) { console.log(jo); }); // Object {id: 123456789012345680, name: "James"}
顯然,這個(gè)對象修改數(shù)值之后再傳回服務(wù)器,就會找不到主鍵,或者更新成錯(cuò)誤的數(shù)據(jù),造成一個(gè)不易發(fā)現(xiàn)的巨大BUG。
解決辦法當(dāng)然是有的,JavaScript處理字符串的能力非常強(qiáng),完全可以把服務(wù)器端的64位整數(shù)處理成字符串類型。不過 Json.NET 默認(rèn)是把 long 處理成 number 類型的,如果要處理成 string 類型,需要自定義一個(gè)JsonConverter。
考慮到用十六進(jìn)制表示的整數(shù)看起來比較整齊,所以定義一個(gè)HexLongConverter來轉(zhuǎn)換long/ulong型數(shù)據(jù)與16進(jìn)制表示的字符串。
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) { // 取得讀到的十六進(jìn)制字符串 string hex = reader.Value as string; // 調(diào)用ToInt64擴(kuò)展將字符串轉(zhuǎn)換成long型 // ToInt64擴(kuò)展方法后附 long v = hex.ToInt64(NumberStyles.HexNumber, 0L); // 將v轉(zhuǎn)換成實(shí)際需要的類型 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; } } }
上面的代碼用到了一個(gè)string的擴(kuò)展方法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; } }
在序列化或反序列化模型的時(shí)候,只需要加入HexLongConverter對象作為參數(shù)即可:
// 序列化 string json = JsonConvert.SerializeObject(model, new HexLongConverter()); // 反序列化 SomeModal model = JsonConvert.DeserializeObject(json, new HexLongConverter));
相關(guān)鏈接:
[Json.NET]
http://json.codeplex.com/
http://james.newtonking.com/json