json.Unmarshal() 反序列化字节流到 interface{} 对象,字段 int/int64 类型出现精度丢失

简介: json.Unmarshal() 反序列化字节流到 interface{} 对象,字段 int/int64 类型出现精度丢失

问题描述

今天遇到一个 json.Unmarshal() 反序列化字节流到 interface{} 对象,int/int64 类型出现精度丢失的问题,记录一下。下面是网上其他同学的类似的代码,跟我的场景很像,所以直接拿过来作为案发现场代码用了。

jsonStr := `{"id":3861708980690657283}`
  result := make(map[string]interface{})
  err := json.Unmarshal([]byte(jsonStr), &result)
  if err != nil {
    fmt.Println(err)
  }
  id := result["id"]
  fmt.Printf("type=%T, val=%v\n",id, id)

输出

type=float64, val=3.861708980690657e+17

反序列化得到的 id 字段变成了 float64 类型,值输出形式也变成了科学计数法形式,这当然难不倒我,略一百度,就查到了 json.Unmarshal() 反序列化字节流到 interface{} 对象时,如果原来是 int 类型,会被反序列化成 float64 类型,网上的解决方案是对 int 字段进行类型强转

id := int64(result["id"].(float64)) // 先强转成 float64类型,再强转成int64 类型
  fmt.Printf("type=%T, val=%v\n",id, id)

输出

type=int, val=3861708980690657280

查看输出结果,可以发现,类型变成了 int 类型,val 表现形式也变成了整数形式,但是出现了精度丢失,原来值是 3861708980690657283,变成了 3861708980690657280,最后一位的精度丢失了。

再次开始网上冲浪,通过参考中提到的 2 篇博客发现了有 2 种解决方案:

解决方案

1、使用 decode+UseNumber()

不使用 json.Unmarshal() 来反序列对象,而是采用 decode+UseNumber() 来实现反序列化。

jsonStr := `{"id":3861708980690657283}`
  result := make(map[string]interface{})
  decoder := json.NewDecoder(bytes.NewBufferString(jsonStr))
  decoder.UseNumber() // 指定使用 Number 类型
  err := decoder.Decode(&result)
  if err != nil {
    fmt.Println(err)
  }
  id := result["id"]
  fmt.Printf("type=%T, val=%v\n",id, id)

输出

type=json.Number, val=3861708980690657283

输出结果精度没有丢失,类型是 json.Number。如果想把 id 转换成 int4 类型,需要先转换成字符串,再强转成 int64 类型

aaa, err := strconv.ParseInt(fmt.Sprintf("%v", id), 10, 64)  // 如果想把 id 转换成 int4 类型,需要先转换成字符串,再强转成 int64 类型
  fmt.Printf("type=%T, val=%v\n",aaa, aaa)  // 输出 type=int64, val=3861708980690657283

2、反序列化的对象改成不是 interface{}, 而是自定义结构体

可以发现上面先后遇到的这两个问题,无论是科学计数法输出,还是精度丢失,都是发生在 Unmarshal反序列化字节流到 interface{} 对象 上,如果如果不是 interface{} 对象,是不会有这个问题的,所以另一种解决方案是,自定义结构体,明确指定每个字段的类型,这样一切都会如你所愿,不会遇到上面任何一个问题。

type Student struct {
    ID int64 `json:"id"`
  }
  jsonStr := `{"id":3861708980690657283}`
  var result Student
  err := json.Unmarshal([]byte(jsonStr), &result)
  if err != nil {
    fmt.Println(err)
  }
  id := result.ID
  fmt.Printf("type=%T, val=%v\n",id, id)

输出

type=int64, val=3861708980690657283

by the way

网友建议前后端在使用 int64 类型进行交互时,尽量将 int64 转成 string 来传输。

参考:[系列] Go - json.Unmarshal 遇到的小坑golang int64 json.Unmarshal精度丢失问题并解决

相关文章
|
9月前
|
Python Windows
[oeasy]python076_int这个词怎么来的_[词根溯源]整数类型_int_integer_touch
本文探讨了“int”一词的起源及其与整数类型的关联。通过词根溯源,揭示“int”来源于“integer”,意为“完整的数”,与零碎的分数相对。同时分析了相关词汇如“tact”(接触)、“touch”(触摸)及衍生词,如“tangential”(切线的)、“intagible”(无形的)和“integral”(完整的、不可或缺的)。文章还结合编程语言特性,解释了Python作为动态类型、强类型语言的特点,并总结了整型变量的概念与意义。最后预告了后续内容,提供了学习资源链接。
326 11
|
9月前
|
存储 C语言 Python
[oeasy]python077_int类型怎么用_整数运算_integer_进制转化_int类
本文主要讲解了Python中`int`类型的应用与特性。首先回顾了`int`词根的溯源,探讨了整型变量的概念及命名规则(如匈牙利命名法)。接着分析了整型变量在内存中的存储位置和地址,并通过`type()`和`id()`函数验证其类型和地址。还介绍了整型变量的运算功能,以及如何通过`int()`函数将字符串转化为整数,支持不同进制间的转换(如二进制转十进制)。此外,文章提及了关键字`del`的使用场景,对比了Python与C语言中`int`的区别,并总结了整型与字符串类型的差异,为后续深入学习奠定基础。
209 1
|
XML JSON 数据可视化
数据集学习笔记(二): 转换不同类型的数据集用于模型训练(XML、VOC、YOLO、COCO、JSON、PNG)
本文详细介绍了不同数据集格式之间的转换方法,包括YOLO、VOC、COCO、JSON、TXT和PNG等格式,以及如何可视化验证数据集。
3482 1
数据集学习笔记(二): 转换不同类型的数据集用于模型训练(XML、VOC、YOLO、COCO、JSON、PNG)
|
JSON 前端开发 JavaScript
聊聊 Go 语言中的 JSON 序列化与 js 前端交互类型失真问题
在Web开发中,后端与前端的数据交换常使用JSON格式,但JavaScript的数字类型仅能安全处理-2^53到2^53间的整数,超出此范围会导致精度丢失。本文通过Go语言的`encoding/json`包,介绍如何通过将大整数以字符串形式序列化和反序列化,有效解决这一问题,确保前后端数据交换的准确性。
346 4
|
Python
[oeasy]python036_数据类型有什么用_type_类型_int_str_查看帮助
本文回顾了Python中`ord()`和`chr()`函数的使用方法,强调了这两个函数互为逆运算:`ord()`通过字符找到对应的序号,`chr()`则通过序号找到对应的字符。文章详细解释了函数参数类型的重要性,即`ord()`需要字符串类型参数,而`chr()`需要整数类型参数。若参数类型错误,则会引发`TypeError`。此外,还介绍了如何使用`type()`函数查询参数类型,并通过示例展示了如何正确使用`ord()`和`chr()`进行转换。最后,强调了在函数调用时正确传递参数类型的重要性。
141 3
【Java基础面试五】、 int类型的数据范围是多少?
这篇文章回答了Java中`int`类型数据的范围是-2^31到2^31-1,并提供了其他基本数据类型的内存占用和数值范围信息。
【Java基础面试五】、 int类型的数据范围是多少?
|
JSON 人工智能 编译器
Go json 能否解码到一个 interface 类型的值
Go json 能否解码到一个 interface 类型的值
123 1
|
数据采集 分布式计算 数据处理
Dataphin常见问题之与指定类型int不兼容如何解决
Dataphin是阿里云提供的一站式数据处理服务,旨在帮助企业构建一体化的智能数据处理平台。Dataphin整合了数据建模、数据处理、数据开发、数据服务等多个功能,支持企业更高效地进行数据治理和分析。
|
SQL 流计算 OceanBase
OceanBase CDC从热OB库采集过来的Tinyint(1)类型会默认转换成Boolean,请教一下,如果想转换成int类型,有什方法么?
【2月更文挑战第25天】OceanBase CDC从热OB库采集过来的Tinyint(1)类型会默认转换成Boolean,请教一下,如果想转换成int类型,有什方法么?
341 3
|
JSON 小程序 JavaScript
小程序根据返回值的int类型渲染不同的状态
小程序根据返回值的int类型渲染不同的状态
218 0