Json.NET特殊处理64位长整型数据
admin
2023-02-15 08:40:05
0

  很多ASP.NET项目,尤其是使用了Ajax的项目,常常需要返回JSON格式的数据。.NET框架从3.5版本开始提供了JSON的序列化和反序列化工具,不过个人感觉不太好用,后来找了第三方的Newtonsoft.Json来用。再后来,在MVC4中,微软已经默认使用Json.NET(Newtonsoft.Json)来处理JSON数据了。


  JavaScript数值精度是32位,如果整数数度超过32位,就会被当作浮点数处理。换句话说,如果从服务端生成的JSON,某个值是64位整数,传到前端JavaScript,再传回服务端,不做任何运算,都可能出现失真。做个实验:


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


  很要命的一点是,数据库设计中常常会用bigint(64位)整数来作为主键,是一个非常重要而且不能有偏差的数据,比如,一个模型:


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


  转换成JSON输出到前端是:


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


  通过Ajax取得的对象输出就有点不妙了


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


  显然,这个对象修改数值之后再传回服务器,就会找不到主键,或者更新成错误的数据,造成一个不易发现的巨大BUG。


  解决办法当然是有的,JavaScript处理字符串的能力非常强,完全可以把服务器端的64位整数处理成字符串类型。不过 Json.NET 默认是把 long 处理成 number 类型的,如果要处理成 string 类型,需要自定义一个JsonConverter。


  考虑到用十六进制表示的整数看起来比较整齐,所以定义一个HexLongConverter来转换long/ulong型数据与16进制表示的字符串。


public class HexLongConverter : JsonConverter
{
    public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
    {
        // 由于CanConvert过滤,数据类型只可能是long或ulong
        // 统一转换成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;
        // 调用ToInt64扩展将字符串转换成long型
        // ToInt64扩展方法后附
        long v = hex.ToInt64(NumberStyles.HexNumber, 0L);
        // 将v转换成实际需要的类型 ulong 或 long(不转换)
        return typeof (ulong) == objectType ? (object) (ulong) v : v;
    }
    public override bool CanConvert(Type objectType)
    {
        // 只处理long和ulong两种类型的数据
        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对象作为参数即可:


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




相关链接:


[Json.NET]

  • http://json.codeplex.com/

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



相关内容

热门资讯

即使中东冲突今天停止,欧盟在未... 新华社布鲁塞尔5月5日电(记者康逸)欧盟能源委员丹·约根森5日警告说,由于伊朗战事以及霍尔木兹海峡遭...
原创 A... 2026 年 4 月,知名产业链分析师郭明錤发布重磅报告:OpenAI正加速研发自研智能手机,以 A...
原创 怎... 你是不是也有过这样的疑惑:明明换了新手机,套餐也升级了,但总感觉网速没快到哪儿去,心里直犯嘀咕——我...
聚力未来通信!成都6G联盟成立 4月28日下午,成都6G联盟成立大会在成都民营经济发展促进中心举行。产业链上下游企业代表、行业专家学...
累计亏损800亿美元,Meta... IT之家 5 月 2 日消息,当地时间 5 月 1 日,据《财富》杂志报道,Meta 第一季度业绩本...
万鑫AI:AI交易大模型驱动下... 在全球金融市场进入高波动、高联动与高智能化并行发展的新阶段后,交易体系的竞争逻辑正在发生明显变化,传...
原创 赛... 2026 年 5 月 5 日,赛力斯股价突发暴跌,盘中跌幅超 6%,市值单日蒸发超百亿,# 赛力斯跌...
杭州博洲电器取得贴片电阻测试治... 国家知识产权局信息显示,杭州博洲电器有限公司取得一项名为“一种贴片电阻测试治具”的专利,授权公告号C...
智聚当下 共启新章——京口区举... 为抢抓人工智能发展新机遇,加快培育新质生产力,推动区域产业数智化转型与高质量发展,4月30日下午,京...
佐林电器取得可调式永磁阻尼器专... 国家知识产权局信息显示,上海佐林电器有限公司取得一项名为“一种可调式永磁阻尼器”的专利,授权公告号C...