Golang JSON 解码时未对接口错误进行处理
来源:stackoverflow
时间:2024-02-26 13:09:24 242浏览 收藏
小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《Golang JSON 解码时未对接口错误进行处理》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!
我是 golang 的菜鸟,非常感谢以下方面的任何帮助
我有这个代码片段,工作正常
var settings cloudsettings type cloudsettings struct { ... a1 *bool `json:"cloud.feature1,omitempty"` ... } err = json.newdecoder(request.body).decode(&settings) an attempt to send an invalid string would raise this error: curl ... -d '{"cloud.feature1" : "junk"}' "message":"error:strconv.parsebool: parsing \"junk\": invalid syntax decoding request body."
现在,我们有一个单独的 localsettings
结构体,并且相同的函数需要有条件地处理云/本地设置解码
所以,代码改为:
var settings interface{} = CloudSettings{} // If the request header says local settings settings = LocalSettings{} /* After this change Decode() no longer raises any error for invalid strings and accepts anything */ err = json.NewDecoder(request.Body).Decode(&settings)
所以问题是为什么我会看到这种行为以及如何解决这个问题?
如果我有 2 个单独的 settings
变量,那么从该点开始的整个代码将被重复,这是我想避免的
正确答案
在第二个片段中,您有一个初始化为结构的接口,但传递该接口的地址。该接口包含 localsettings
或 cloudsetting
值,该值无法被覆盖,因此解码器创建一个 map[string]interface{}
,将传递的接口的值设置为指向该值,并解组数据。当您运行第二个代码段时,您不会初始化本地设置或云设置。
更改:
settings=&cloudsettings{}
或
settings=&localsettings{}
和
err = json.newdecoder(request.body).decode(settings)
它应该按预期运行
根据您的问题,我假设所有字段(即使是具有相同名称的字段)在 json 标记中都有 cloud.
或 local.
前缀。如果是这种情况,您可以简单地将两个选项嵌入到一个类型中:
type wrapper struct { *cloudsettings *localsettings }
然后解组为该包装类型。 json 标记将确保填充正确设置类型的正确字段:
wrapper := &wrapper{} if err := json.newdecoder(request.body).decode(&wrapper); err != nil { // handle } // now to work out which settings were passed: if wrapper.cloudsettings == nil { fmt.println("local settings provided!") // use wrapper.cloudsettings } else { fmt.println("cloud settings provided!") // use wrapper.localsettings }
Playground demo
您提到我们希望看到基于标头值加载的本地设置。您可以简单地解组有效负载,然后检查标头是否与加载的设置类型匹配。如果标头指定了本地设置,但负载包含云设置,则只需返回错误响应。
不过,我在这里假设您的 json 标签对于两种设置类型都是不同的。这并不总是适用,所以如果我的假设不正确,并且某些字段共享相同的 json 标签,那么自定义 unmarshal 函数将是可行的方法:
func (w *wrapper) unmarshaljson(data []byte) error { // say we want to prioritise local over cloud: l := localsettings{} if err := json.unmarshal(data, &l); err == nil { // we could unmarshal into local without a hitch? w.cloudsettings = nil // ensure this is blanked out w.localsettings = &l // set local return nil } // we should use cloud settings c := cloudsettings{} if err := json.unmarshal(data, &c); err != nil { return err } w.localsettings = nil w.cloudsettings = &c return nil }
这样,任何冲突都会得到解决,并且我们可以控制哪些设置优先。同样,无论 json 解组的结果如何,您都可以简单地交叉检查标头值 + 填充的设置类型,然后从那里获取它。
最后,如果两种设置类型之间存在相当大的重叠,您也可以将有效负载解组为两种类型,并填充包装类型中的两个字段:
func (w *Wrapper) UnmarshalJSON(data []byte) error { *w = Wrapper{} // make sure we start with a clean slate l := LocalSettings{} var localFail err if err := json.Unmarshal(data, &l); err == nil { w.LocalSettings = &l // set local } else { localFail = err } c := CloudSettings{} if err := json.Unmarshal(data, &c); err == nil { w.CloudSettings = &c } else if localFail != nil { // both unmarshal calls failed return err // maybe wrap/return custom error } return nil // one or more unmarshals were successful }
这应该可以解决问题
以上就是《Golang JSON 解码时未对接口错误进行处理》的详细内容,更多关于的资料请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习