首页 > 文章列表 > 在 UnmarshalJSON 方法中递归调用 json.Unmarshal 不会引起堆栈溢出情况

在 UnmarshalJSON 方法中递归调用 json.Unmarshal 不会引起堆栈溢出情况

106 2024-02-22
问题内容

我想执行一些额外的步骤来初始化我的实现 UnmarshalJSON 中的数据结构。在该实现中调用 json.Unmarshal(b, type) 自然会导致堆栈溢出。

JSON 解码器不断尝试查找是否有自定义 UnmarshalJSON 实现,然后再次调用 json.Unmarshal

还有其他方法可以做到这一点吗?只需调用底层默认实现就不会导致此问题?


正确答案


避免这种情况/防止这种情况的一种简单而常见的方法是使用 type< 创建一个新类型/a> 关键字,并使用类型 conversion 来传递该类型的值(该值可以如果是您的原始值,则可以进行类型转换,因为新类型将原始类型作为其基础类型)。

这是有效的,因为 type 关键字创建了一个新类型,并且新类型将具有零个方法(它不会“继承”基础类型的方法)。

这会产生一些运行时开销吗?否。引用自 规范:转换:

让我们看一个例子。我们有一个带有数字 AgePerson 类型,并且我们要确保 Age 不能为负数(小于 0)。

type Person struct {
    Name string `json:"name"`
    Age  int    `json:"age"`
}

func (p *Person) UnmarshalJSON(data []byte) error {
    type person2 Person
    if err := json.Unmarshal(data, (*person2)(p)); err != nil {
        return err
    }

    // Post-processing after unmarshaling:
    if p.Age < 0 {
        p.Age = 0
    }
    return nil
}

测试它:

var p *Person
fmt.Println(json.Unmarshal([]byte(`{"name":"Bob","age":10}`), &p))
fmt.Println(p)

fmt.Println(json.Unmarshal([]byte(`{"name":"Bob","age":-1}`), &p))
fmt.Println(p)

输出(在 Go Playground 上尝试):

<nil>
&{Bob 10}
<nil>
&{Bob 0}

当然,相同的技术也适用于自定义封送处理(MarshalJSON()):

func (p *Person) MarshalJSON() ([]byte, error) {
    // Pre-processing before marshaling:
    if p.Age < 0 {
        p.Age = 0
    }

    type person2 Person
    return json.Marshal((*person2)(p))
}

测试它:

p = &Person{"Bob", 10}
fmt.Println(json.NewEncoder(os.Stdout).Encode(p))
p = &Person{"Bob", -1}
fmt.Println(json.NewEncoder(os.Stdout).Encode(p))

输出(在同一个 Go Playground 示例中):

{"name":"Bob","age":10}
<nil>
{"name":"Bob","age":0}
<nil>

一个非常相似的问题是,当您为 fmtString() string 方法时a> 包,并且您想要使用您修改的默认字符串表示形式。在这里阅读更多相关信息:t 和 *t 之间的区别