Go JSON Unmarshal 嵌套对象为原始字节或字符串

go json unmarshal 嵌套对象为原始字节或字符串

本文深入探讨了在Go语言中使用`encoding/json`包处理JSON数据时,如何将嵌套的JSON对象直接解组(Unmarshal)为原始字节切片(`[]byte`)或字符串,而非将其解析为具体的Go结构体。通过介绍`json.RawMessage`类型,文章详细阐述了其原理、用法,并提供了实用的代码示例,帮助开发者有效地处理复杂的JSON结构,实现按需延迟解析,避免不必要的预解析错误。

在Go语言中,处理JSON数据是日常开发中常见的任务。encoding/json包提供了强大的功能来序列化(Marshal)和反序列化(Unmarshal)JSON。然而,在某些场景下,我们可能不希望立即解析JSON中所有的嵌套对象,而是希望将某个特定的嵌套对象作为一个原始的JSON字符串或字节切片来处理。例如,当JSON结构非常复杂、部分字段不确定或仅在特定条件下才需要进一步解析时,这种需求尤为突出。

问题场景:嵌套JSON对象解组为原始数据

假设我们有如下JSON数据:

{    "id"  : 15,    "foo" : { "foo": 123, "bar": "baz" }}

我们希望将其解组到一个Go结构体中,其中id字段映射为int64,而foo字段(一个嵌套的JSON对象)则直接作为原始的字节切片([]byte)存储,而不是被解析成另一个Go结构体。

如果尝试将foo字段定义为[]byte类型:

type Bar struct {    ID  int64  `json:"id"`    Foo []byte `json:"foo"`}

当执行json.Unmarshal时,会遇到类似以下的错误:

json: cannot unmarshal object into Go value of type []uint8

这个错误明确指出,encoding/json无法将一个JSON对象直接解组到[]byte类型中,因为它期望的是一个JSON数组或字符串。为了解决这个问题,我们需要利用encoding/json包提供的一个特殊类型:json.RawMessage。

解决方案:使用 json.RawMessage

json.RawMessage是encoding/json包中提供的一个特殊类型,它的底层实际上是[]byte。它实现了json.Marshaler和json.Unmarshaler接口,这使得它在JSON编解码过程中具有特殊的行为。

json.RawMessage的特性:

延迟解码: 当json.Unmarshal遇到json.RawMessage类型的字段时,它不会尝试解析该字段的内部结构,而是直接将其对应的原始JSON数据(包括大括号、引号等)作为字节切片存储。预计算编码 同样,当json.Marshal遇到json.RawMessage类型的字段时,它会直接将存储在其中的字节切片作为JSON的一部分输出,而不会再次对其进行编码。

这使得json.RawMessage非常适合用于存储未解析的JSON片段,或者在需要时手动进行二次解析。

示例代码

下面是一个使用json.RawMessage来解决上述问题的完整示例:

package mainimport (    "encoding/json"    "fmt")// 假设的JSON数据var jsonStr = []byte(`{    "id"  : 15,    "foo" : { "foo": 123, "bar": "baz" }}`)// 定义目标结构体,使用json.RawMessage来存储嵌套的"foo"对象type Bar struct {    ID  int64           `json:"id"`    Foo json.RawMessage `json:"foo"` // 使用json.RawMessage}func main() {    var bar Bar    // 执行JSON解组    err := json.Unmarshal(jsonStr, &bar)    if err != nil {        fmt.Printf("Unmarshal error: %vn", err)        return    }    // 打印解组后的结构体    fmt.Printf("解组结果: %+vn", bar)    // 进一步解析嵌套的"foo"字段    // 如果需要,可以随时将RawMessage中的内容解析到另一个结构体    type FooDetail struct {        Foo int    `json:"foo"`        Bar string `json:"bar"`    }    var fooDetail FooDetail    err = json.Unmarshal(bar.Foo, &fooDetail) // 对bar.Foo进行二次Unmarshal    if err != nil {        fmt.Printf("Unmarshal fooDetail error: %vn", err)        return    }    fmt.Printf("Foo字段二次解析结果: %+vn", fooDetail)}

代码解释:

type Bar struct { … Foo json.RawMessagejson:”foo”… }: 在Bar结构体中,我们将Foo字段的类型定义为json.RawMessage。通过json:”foo”标签,它将与输入JSON中的”foo”键进行匹配。err := json.Unmarshal(jsonStr, &bar): 当执行Unmarshal时,encoding/json包会识别到Foo字段是json.RawMessage类型。因此,它不会尝试解析{“foo”: 123, “bar”: “baz”}这个对象,而是将其原始的字节表示(包括大括号和内部内容)直接存储到bar.Foo中。fmt.Printf(“解组结果: %+vn”, bar): 打印结果会显示bar.Foo字段包含的是原始的字节切片,例如Foo:[123 32 34 102 111 111 34 58 32 49 50 51 44 32 34 98 97 114 34 58 32 34 98 97 122 34 32 125],这实际上就是{“foo”: 123, “bar”: “baz”}的ASCII字节表示。二次解析 (json.Unmarshal(bar.Foo, &fooDetail)): 示例中还展示了如何对bar.Foo(即json.RawMessage)进行二次Unmarshal。这允许我们在需要时,将原始的JSON片段解析成一个具体的Go结构体,实现了按需解析。

注意事项

错误处理: 在实际应用中,务必对json.Unmarshal等操作进行严格的错误检查,确保数据处理的健壮性。性能考量: json.RawMessage避免了立即解析整个嵌套结构,这在处理大型JSON或仅需要部分字段的场景下,可以带来性能上的优势。与string的区别 虽然json.RawMessage底层是[]byte,但它不是普通的[]byte。如果直接将字段类型定义为string或[]byte,json.Unmarshal会尝试将其作为JSON字符串或JSON数组(分别对应Go的string和[]byte)进行解析,而不是一个JSON对象。json.RawMessage的特殊之处在于它能够“捕获”任何有效的JSON片段(对象、数组、字符串、数字、布尔值、null)作为原始字节。

总结

json.RawMessage是Go语言encoding/json包中一个非常实用的类型,它为开发者提供了一种灵活的方式来处理复杂的JSON结构。通过将嵌套的JSON对象解组为原始字节切片,我们可以实现延迟解析,根据业务逻辑按需处理数据,从而提高程序的灵活性和效率。掌握json.RawMessage的用法,是Go语言JSON处理进阶的关键一步。

以上就是Go JSON Unmarshal 嵌套对象为原始字节或字符串的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1422042.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月16日 16:30:48
下一篇 2025年12月16日 16:31:05

相关推荐

  • c语言与go语言的区别是什么

    区别:1、C语言源文件的扩展名是“.h”和“.c”,Go语言源文件的扩展名是“.go”。2、C语言中通过文件来管理代码,Go语言中通过包来管理代码。3、C语言中一共有32个关键字,Go语言中一共有25个关键字。 本教程操作环境:windows7系统、c99&&GO 1.18版本、De…

    2025年12月17日 好文分享
    000
  • unsigned int几个字节

    unsigned int几个字节 C语言中unsigned int代表无符号整型。并没有确定规定它占用几个字节,具体是由编译器来决定的,例如Visual C++规定unsigned int占4字节,Turbo 2.0中,规定unsigned int占2字节,也就是说int可以占用2字节也可以占用4字…

    2025年12月17日
    000
  • i++和++i的区别及举例说明

    i++和++i的区别及举例说明 i++和++i命令的区别有: 1、赋值顺序不同 ++ i 是先加后赋值;i ++ 是先赋值后加;++i和i++都是分两步完成的。 因为++i 是后面一步才赋值的,所以它能够当作一个变量进行级联赋值,++i = a =b,即 ++i 是一个左值;i++ 的后面一步是自增…

    2025年12月17日
    000
  • scanf和getchar的区别

    scanf和getchar的区别 一、函数格式不同 scanf函数是格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量中。 getchar函数是键盘输入函数,其功能是从键盘上输入一个字符。 二、读取方式不同 scanf函数在读取数字时会跳过空格、制表符和换行符。 getchar函数只能输…

    2025年12月17日
    000
  • #ifndef和#define的区别

    #ifndef和#define的区别 一、使用场景不同: #ifndef使用场景为: 1、头文件中使用,防止头文件被多重调用。 2、作为测试使用,省去注释代码的麻烦。 3、作为不同角色或者场景的判断使用。 #define使用场景: 宏定义 二、含义不同: #ifndef表示ifnotdefine。 …

    2025年12月17日
    000
  • printf和scanf的区别

    printf和scanf的区别 ● 这是两个功能完全不同的函数,printf向标准输出设备(一般是显示器)输出数据,scanf从标准输入设备(一般是键盘)输入数据。 ● printf是输出函数,scanf是输入函数。 拓展内容: printf()函数: 是格式化输出函数, 一般用于向标准输出设备按规…

    2025年12月17日
    000
  • .NetCore如何获取Json和Xml格式的配置信息讲解

    本篇将和大家分享的是如何获取json和xml格式的配置信息,主要介绍的是configuration扩展方法的使用,对.netcore 获取json和xml格式的配置信息的相关知识,感兴趣的朋友一起看看吧 本篇将和大家分享的是:如何获取Json和Xml格式的配置信息,主要介绍的是Configurati…

    2025年12月17日 好文分享
    000
  • 比较TCP与UDP之间的区别

    tcp(传输控制协议): 1)提供ip环境下的数据可靠传输(一台计算机发出的字节流会无差错的发往网络上的其他计算机,而且计算机a接收数据包的时候,也会向计算机b回发数据包,这也会产生部分通信量),有效流控,全双工操作(数据在两个方向上能同时传递),多路复用服务,是面向连接,端到端的传输; 2)面向连…

    好文分享 2025年12月17日
    000
  • 比较C#中值类型和引用类型的区别

    clr支持两种类型:值类型和引用类型,看起来fcl的大多数类型是引用类型,但用的最多的还是值类型。引用类型总是从托管堆中分配,在用new操作符实例一个对象,返回对象内存地址存放在一个变量中。在使用引用类型时要了解其四个心理因素:        1.内存必须从托管堆中分配        2.堆上分配的…

    好文分享 2025年12月17日
    000
  • 关于json result的实例代码

    public jsonresult jsondata()        {            httpcontext.response.appendheader(“access-control-allow-origin”, “*”);       …

    好文分享 2025年12月17日
    000
  • C# 将 Json 解析成 DateTable

    c# 将 json 解析成 datetable  #region 将 Json 解析成 DateTable /// /// 将 Json 解析成 DateTable。 /// Json 数据格式如: /// {table:[{column1:1,column2:2,column3:3},{colum…

    2025年12月17日
    000
  • C# Json 序列化与反序列化一

    public class JsonSerializer { /// /// json序列化 /// /// /// /// public static string JsonStringSerializer(T t) { DataContractJsonSerializer ser = new Da…

    好文分享 2025年12月17日
    000
  • C# web api返回类型设置为json的两种方法

    web api写api接口时默认返回的是把你的对象序列化后以xml形式返回,那么怎样才能让其返回为json呢,下面就介绍两种方法: 方法一:(改配置法)  找到global.asax文件,在application_start()方法中添加一句:  GlobalConfiguration.Config…

    好文分享 2025年12月17日
    000
  • XML中如何压缩文件_XML压缩XML文件的方法与技巧

    答案:通过ZIP/GZIP压缩、优化XML结构、使用EXI等专用格式可显著减小XML文件体积。具体包括利用通用算法压缩、精简标签与属性、采用二进制交换格式,并结合场景选择兼顾压缩率与兼容性的方案。 处理XML文件时,文件体积过大常常影响传输效率和存储成本。通过合理的压缩方法,可以显著减小XML文件的…

    2025年12月17日
    000
  • 什么是XML Infoset

    XML Infoset是W3C定义的抽象数据模型,用于标准化XML文档解析后的信息表示。它定义了11种信息项(如文档、元素、属性等),屏蔽物理格式差异,确保不同解析器对XML内容的理解一致。DOM和SAX等解析技术均基于Infoset构建:DOM将其具象化为树结构,SAX则通过事件流式暴露信息项。I…

    2025年12月17日
    000
  • RSS订阅中的作者信息格式

    RSS和Atom中作者信息通过或标签标识,包含姓名、邮箱及网站链接,支持多作者;正确设置有助于提升内容可信度、便于追踪与SEO。 RSS订阅中的作者信息格式,主要用于标识文章的作者,让读者知道是谁写的,方便追踪特定作者的内容。格式通常包含作者姓名、邮箱,有时还会包含作者的网站链接。 作者信息的常见格…

    2025年12月17日
    000
  • XML中如何获取根节点属性_XML获取根节点属性的操作步骤

    XML根节点有且仅有一个,可包含属性;2. Python用ET.parse解析,root.get(“属性名”)获取属性值;3. JavaScript用DOMParser解析,xmlDoc.documentElement获取根节点,getAttribute读取属性;4. Jav…

    2025年12月17日
    000
  • XML中如何提取指定节点_XML提取指定节点的详细步骤

    首先理解XML结构,明确目标节点路径;接着使用XPath表达式如//title或/books/book[@id=’1′]定位节点;然后通过Python的lxml库解析XML并执行XPath提取文本或属性;最后处理多层级节点与属性,结合条件筛选和遍历方法精准获取数据。 在处理X…

    2025年12月17日
    000
  • XML中如何比较XML文件差异_XML比较XML文件差异的操作方法

    使用专业工具或编程方法可精准比对XML差异。XMLSpy和Oxygen提供可视化比对,DiffNow适合在线轻量比对;Python的ElementTree、Java的XMLUnit支持代码级控制;xmldiff命令行工具便于自动化;预处理需统一格式、忽略无关差异,关注命名空间与大文件性能,根据场景选…

    2025年12月17日
    000
  • XML中如何解压XML字符串_XML解压XML字符串的操作方法

    先解压再解析XML。C#用GZipStream解压字节流并转字符串,Java用GZIPInputStream或InflaterInputStream读取压缩数据,结合StreamReader或BufferedReader还原为明文XML后,交由XDocument或DocumentBuilder解析;…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信