目录
问题内容
解决方法
首页 后端开发 Golang 在 UnmarshalJSON 函数内调用 json.Unmarshal 不会导致堆栈溢出

在 UnmarshalJSON 函数内调用 json.Unmarshal 不会导致堆栈溢出

Feb 12, 2024 am 08:27 AM
堆栈溢出 字符串解析 overflow

在 UnmarshalJSON 函数内调用 json.Unmarshal 不会导致堆栈溢出

在处理 JSON 数据时,我们通常会使用 json.Unmarshal 函数将 JSON 字符串解析为 Go 语言中的结构体。然而,在 UnmarshalJSON 函数内部调用 json.Unmarshal 函数可能会导致堆栈溢出的错误。这是因为 UnmarshalJSON 函数在解析 JSON 数据时会再次调用自身,从而导致无限循环。为了避免这种情况,我们可以使用 json.Decoder 的 Decode 方法来解析 JSON 数据,而不是直接调用 json.Unmarshal 函数。这样做可以确保不会造成堆栈溢出的问题,保证代码的健壮性和性能。

问题内容

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

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

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

解决方法

避免这种情况/防止这种情况的一种简单而常见的方法是使用 type 关键字,并使用类型 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>
登录后复制

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

以上是在 UnmarshalJSON 函数内调用 json.Unmarshal 不会导致堆栈溢出的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
3 周前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解锁Myrise中的所有内容
3 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

2018-2024年比特币最新价格美元大全 2018-2024年比特币最新价格美元大全 Feb 15, 2025 pm 07:12 PM

实时比特币美元价格 影响比特币价格的因素 预测比特币未来价格的指标 以下是 2018-2024 年比特币价格的一些关键信息:

如何通过CSS自定义resize符号并使其与背景色统一? 如何通过CSS自定义resize符号并使其与背景色统一? Apr 05, 2025 pm 02:30 PM

CSS自定义resize符号的方法与背景色统一在日常开发中,我们经常会遇到需要自定义用户界面细节的情况,比如调...

如何使用CSS的clip-path属性实现分段器的45度曲线效果? 如何使用CSS的clip-path属性实现分段器的45度曲线效果? Apr 04, 2025 pm 11:45 PM

如何实现分段器的45度曲线效果?在实现分段器的过程中,如何让点击左侧按钮时右侧边框变成45度曲线,而点�...

Flex布局下文字超出省略却撑开容器?如何解决? Flex布局下文字超出省略却撑开容器?如何解决? Apr 05, 2025 pm 11:00 PM

Flex布局下文字超出省略导致容器撑开的问题及解决方法在使用Flex...

H5页面制作是前端开发吗 H5页面制作是前端开发吗 Apr 05, 2025 pm 11:42 PM

是的,H5页面制作是前端开发的重要实现方式,涉及HTML、CSS和JavaScript等核心技术。开发者通过巧妙结合这些技术,例如使用&lt;canvas&gt;标签绘制图形或使用JavaScript控制交互行为,构建出动态且功能强大的H5页面。

比特币诞生至今历史价格明细最全汇总(2025年最新版) 比特币诞生至今历史价格明细最全汇总(2025年最新版) Feb 15, 2025 pm 06:45 PM

比特币历史价格重要节点2009 年1 月 3 日:创世区块生成,第一个比特币产生,价值为 0 美元。10 月 5 日:第一笔比特币交易,一名程序员用 10,000 比特币购买了两个披萨,相当于 0.008 美元。2010 年2 月 9 日:Mt. Gox 交易所上线,成为比特币早期交易的主要平台。5 月 22 日:比特币首次突破 1 美元。7 月 17 日:比特币价格暴跌至 0.008 美元,触及历史低点。2011 年2 月 9 日:比特币价格首次突破 10 美元。4 月 10 日:Mt. Go

ChatGPT时代,技术问答社区思否如何应对挑战? ChatGPT时代,技术问答社区思否如何应对挑战? Apr 01, 2025 pm 11:51 PM

ChatGPT时代的技术问答社区:思否(SegmentFault)的应对策略StackOverflow...

为什么inline-block元素会出现错位现象?如何解决这个问题? 为什么inline-block元素会出现错位现象?如何解决这个问题? Apr 04, 2025 pm 10:39 PM

关于inline-block元素错位显示的原因及解决方案在编写网页布局时,我们常常会遇到一些看似奇怪的显示问题。比...

See all articles