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 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习