0

0

在 Go 中维护未解析的 JSON 字段的最佳方法

花韻仙語

花韻仙語

发布时间:2025-10-29 14:31:13

|

941人浏览过

|

来源于php中文网

原创

在 go 中维护未解析的 json 字段的最佳方法

本文探讨了在 Go 语言中使用 `encoding/json` 包处理 JSON 数据时,如何保留结构体中未定义的、动态的 JSON 字段。我们将介绍使用 `json.RawMessage` 类型以及自定义 `Unmarshaler` 和 `Marshaler` 接口的两种方法,以便在解码和编码 JSON 数据时,能够灵活地处理未知字段,从而避免完全使用 `map[string]interface{}` 带来的复杂性。

在 Go 语言中处理 JSON 数据时,我们经常会遇到需要将 JSON 数据解码到结构体中,进行一些操作,然后再将修改后的数据编码回 JSON 格式的需求。然而,JSON 数据中可能包含一些我们预先不知道或者不需要在结构体中定义的字段。如果我们简单地使用 json.Unmarshal 将 JSON 数据解码到结构体中,那么这些未定义的字段将会丢失。本文将介绍几种在 Go 中维护这些未解析 JSON 字段的方法,以便在重新编码时能够保留这些字段。

使用 json.RawMessage

json.RawMessage 类型允许我们将 JSON 数据的一部分表示为原始的 JSON 数据,而不进行具体的解码。这使得我们可以在结构体中定义一个 json.RawMessage 类型的字段,用于存储我们不关心的 JSON 部分。

例如,假设我们有以下 JSON 数据:

{
  "name": "Joe Smith",
  "age": 42,
  "phone": "614-555-1212",
  "debug": true,
  "codeword": "wolf"
}

我们只想处理 name、age 和 address 字段,而保留 debug 和 codeword 字段。我们可以定义如下结构体:

package main

import (
    "encoding/json"
    "fmt"
)

type Person struct {
    Name    string          `json:"name"`
    Age     uint            `json:"age"`
    Address json.RawMessage `json:"address"` // Store unknown fields
}

func main() {
    jsonData := []byte(`{ "name": "Joe Smith", "age": 42, "address": { "phone": "614-555-1212", "debug": true, "codeword": "wolf" }}`)

    var p Person
    err := json.Unmarshal(jsonData, &p)
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    fmt.Printf("Name: %s, Age: %d\n", p.Name, p.Age)

    // Happy birthday
    p.Age++

    data, err := json.Marshal(p)
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    fmt.Println(string(data))
}

在这个例子中,Address 字段是一个 json.RawMessage 类型,它会存储 JSON 数据中 address 字段的原始 JSON 数据。当我们使用 json.Marshal 重新编码结构体时,Address 字段中的原始 JSON 数据会被保留。

输出:

SEEK.ai
SEEK.ai

AI驱动的智能数据解决方案,询问您的任何数据并立即获得答案

下载
Name: Joe Smith, Age: 42
{"name":"Joe Smith","age":43,"address":{"phone": "614-555-1212", "debug": true, "codeword": "wolf"}}

注意事项:

  • json.RawMessage 实际上是一个 []byte 类型,因此在使用它时需要注意类型转换。
  • json.RawMessage 存储的是原始的 JSON 数据,因此在处理时需要小心,避免出现 JSON 格式错误。
  • json.RawMessage 适用于存储嵌套的 JSON 对象或数组,对于简单的键值对,使用 map[string]interface{} 可能更方便。

使用自定义 Unmarshaler 和 Marshaler 接口

另一种方法是实现自定义的 Unmarshaler 和 Marshaler 接口。Unmarshaler 接口允许我们自定义 JSON 解码的行为,Marshaler 接口允许我们自定义 JSON 编码的行为。

通过实现这两个接口,我们可以在解码时将未知的字段存储到一个 map[string]interface{} 类型的字段中,然后在编码时将这些字段重新添加到 JSON 数据中。

package main

import (
    "encoding/json"
    "fmt"
)

type Person struct {
    Name    string                 `json:"name"`
    Age     uint                   `json:"age"`
    Phone   string                 `json:"phone"`
    UnknownFields map[string]interface{} `json:"-"` // Ignore during standard marshaling
}

func (p *Person) UnmarshalJSON(data []byte) error {
    // Define a type to avoid recursion
    type Alias Person
    aux := &Alias{}

    if err := json.Unmarshal(data, &aux); err != nil {
        return err
    }

    *p = Person(*aux)

    // Unmarshal to a map to capture unknown fields
    var m map[string]interface{}
    if err := json.Unmarshal(data, &m); err != nil {
        return err
    }

    // Remove known fields from the map
    delete(m, "name")
    delete(m, "age")
    delete(m, "phone")

    // Store unknown fields
    p.UnknownFields = m

    return nil
}

func (p Person) MarshalJSON() ([]byte, error) {
    // Define a type to avoid recursion
    type Alias Person
    aux := &Alias{
        Name: p.Name,
        Age:  p.Age,
        Phone: p.Phone,
    }

    data, err := json.Marshal(aux)
    if err != nil {
        return nil, err
    }

    var m map[string]interface{}
    if err := json.Unmarshal(data, &m); err != nil {
        return nil, err
    }

    // Add unknown fields to the map
    for k, v := range p.UnknownFields {
        m[k] = v
    }

    return json.Marshal(m)
}

func main() {
    jsonData := []byte(`{ "name": "Joe Smith", "age": 42, "phone": "614-555-1212", "debug": true, "codeword": "wolf" }`)

    var p Person
    err := json.Unmarshal(jsonData, &p)
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    fmt.Printf("Name: %s, Age: %d, Phone: %s\n", p.Name, p.Age, p.Phone)

    // Happy birthday
    p.Age++

    data, err := json.Marshal(p)
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    fmt.Println(string(data))
}

输出:

Name: Joe Smith, Age: 42, Phone: 614-555-1212
{"age":43,"codeword":"wolf","debug":true,"name":"Joe Smith","phone":"614-555-1212"}

注意事项:

  • 在实现 UnmarshalJSON 和 MarshalJSON 接口时,需要注意避免无限递归调用。
  • UnknownFields 字段使用了 json:"-" 标签,这意味着在默认的编码过程中,该字段会被忽略。
  • 这种方法需要编写更多的代码,但提供了更大的灵活性,可以自定义解码和编码的行为。

总结

本文介绍了两种在 Go 语言中维护未解析 JSON 字段的方法:使用 json.RawMessage 和使用自定义 Unmarshaler 和 Marshaler 接口。选择哪种方法取决于具体的需求和场景。如果只需要保留部分 JSON 数据,json.RawMessage 是一个简单有效的选择。如果需要更灵活地处理 JSON 数据,可以考虑实现自定义的 Unmarshaler 和 Marshaler 接口。无论选择哪种方法,都需要仔细考虑 JSON 数据的结构和字段,以便正确地处理未知字段。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

419

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

240

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

192

2025.07.04

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1133

2023.10.19

java入门学习合集
java入门学习合集

本专题整合了java入门学习指南、初学者项目实战、入门到精通等等内容,阅读专题下面的文章了解更多详细学习方法。

1

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.6万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号