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精度丢失问题并解决

相关文章
|
2天前
|
存储 JSON 数据处理
从JSON数据到Pandas DataFrame:如何解析出所需字段
从JSON数据到Pandas DataFrame:如何解析出所需字段
14 1
|
2天前
|
JSON 数据格式
JSON对象相互转换
JSON对象相互转换
9 3
|
2天前
|
XML 存储 JSON
c#XML、JSON的序列化和反序列化,看完你就懂了
c#XML、JSON的序列化和反序列化,看完你就懂了
28 0
|
2天前
|
JSON Java Linux
【探索Linux】P.30(序列化和反序列化 | JSON序列化库 [ C++ ] )
【探索Linux】P.30(序列化和反序列化 | JSON序列化库 [ C++ ] )
21 2
|
2天前
|
SQL DataWorks 关系型数据库
DataWorks操作报错合集之DataWorks在同步mysql时报错Code:[Framework-02],mysql里面有个json类型字段,是什么原因导致的
DataWorks是阿里云提供的一站式大数据开发与治理平台,支持数据集成、数据开发、数据服务、数据质量管理、数据安全管理等全流程数据处理。在使用DataWorks过程中,可能会遇到各种操作报错。以下是一些常见的报错情况及其可能的原因和解决方法。
30 0
|
2天前
|
分布式计算 DataWorks 关系型数据库
DataWorks产品使用合集之在DataWorks中,使用JSON解析函数将MySQL表中的字段解析成多个字段将这些字段写入到ODPS(MaxCompute)中如何解决
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
29 3
|
2天前
|
存储 JSON DataWorks
DataWorks产品使用合集之DataWorks将 MongoDB 中的数组类型写入到 DataWorks 的单个字段时,表示为字符串格式而非 JSON 格式如何解决
DataWorks作为一站式的数据开发与治理平台,提供了从数据采集、清洗、开发、调度、服务化、质量监控到安全管理的全套解决方案,帮助企业构建高效、规范、安全的大数据处理体系。以下是对DataWorks产品使用合集的概述,涵盖数据处理的各个环节。
26 3
|
2天前
|
前端开发
【专栏】在前端开发中,package.json 文件是项目的重要配置文件,其中包含了许多与项目相关的信息和设置
【4月更文挑战第29天】`package.json`的`proxy`字段用于配置开发环境中的代理服务器,解决跨域问题并模拟后端响应。它是字符串类型,值为代理服务器地址。主要应用场景包括前端跨域请求和本地调试。配置时在`package.json`顶层添加`proxy`字段,如`"proxy": "http://localhost:8080"`。该配置仅在开发环境中生效,生产环境需另寻解决方案。
|
2天前
|
JSON Java 数据处理
Spring Boot与Jsonson对象:灵活的JSON操作实战
【4月更文挑战第28天】在现代Web应用开发中,JSON数据格式的处理至关重要。假设 "Jsonson" 代表一个类似于Jackson的库,这样的工具在Spring Boot中用于处理JSON。本篇博客将介绍Spring Boot中处理JSON数据的基本概念,并通过实际例子展示如何使用类似Jackson的工具进行数据处理。
27 0
|
2天前
|
小程序 开发者
【微信小程序】微信开发者工具 app.json: [“subpackages“][0][“root“] 字段需为目录 已解决
【微信小程序】微信开发者工具 app.json: [“subpackages“][0][“root“] 字段需为目录 已解决
14 0

热门文章

最新文章