序列化反序列时精度丢失的避坑说明

 1. 背景
        在双方通过HTTP网络协议进行数据交互时,发送方往往会将Json对象数据序列化为字符串(ObjectMapper.writeValueAsString(Object value)),而接收方也往往会将接收到字符串数据反序列化为Json对象(ObjectMapper.readValue(String content, Classe valueType))。其中,当Json对象内出现数值类型,接收方在readValue时,数值字段会出现科学计数法与精度丢失问题。

        只要涉及到序列化和反序列化的框架都可能出现该问题,以下使用jackson框架作为演示讲解。

        2. 场景复现
a. 当数值类型是浮点数且达到千万级别时,字符串转Json对象会出现科学计数法问题

b. 当数值类型是浮点数且以“0”结尾时,字符串转Json对象会“抹0”问题

         3. 解决方案
交互双方约定数值类型的数据也采用字符串传输,字符串传输不存在以上两种问题。
针对出现科学计数法问题,可对ObjectMapper开启反序列功能的USE_BIG_DECIMAL_FOR_FLOATS配置,该配置默认是关闭的。
配置:


private static mapper;
static{
    mapper = new ObjectMapper();
    mapper.enable(DeserializationFeature.USE_BIG_DECIMAL_FOR_FLOATS);
}


验证:

针对精度丢失("抹0")问题,同样也可以开启配置

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在C#中,`IntPtr`是用于表示无符号32位整数的类型,但它是不可序列化的,因为它不是.NET框架内置的可序列化类型。如果你需要将`IntPtr`序列化序列化,通常的做法是先将其换为其他可序列化的类型(如`int`或`long`),然后再进行序列化序列化。 以下是一个简单的示例,如何使用`BinaryFormatter`来序列化序列化`int`: ```csharp using System; using System.IO; using System.Runtime.Serialization.Formatters.Binary; public class Program { public static void Main() { // 创建一个IntPtr实例 IntPtr intptrValue = new IntPtr(123456789); // 将IntPtr换为int int intValue = unchecked((int)intptrValue); // 创建一个内存流用于序列化 MemoryStream memoryStream = new MemoryStream(); BinaryFormatter formatter = new BinaryFormatter(); try { // 序列化int到流中 formatter.Serialize(memoryStream, intValue); memoryStream.Seek(0, SeekOrigin.Begin); // 重置流位置以便读取 // 序列化流回int object deserializedValue = formatter.Deserialize(memoryStream); int deserializedInt = (int)deserializedValue; Console.WriteLine("Original IntPtr: " + intptrValue); Console.WriteLine("Deserialized Int: " + deserializedInt); } finally { // 关闭流 memoryStream.Close(); } } } ``` 请注意,这种方法可能会丢失`IntPtr`的原始精度(如果数值超过`int`的范围),并且`BinaryFormatter`的性能也不是最优。如果你的数据需要更高级别的控制,可能需要考虑使用自定义序列化或第三方库(如Json.NET)来处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值