将 JSON 解组到 Go 结构体时,辨别字段是空还是不存在可能是一个挑战。这两种情况都会导致结构体中出现 nil 值,从而很难确定该字段的原始存在。
为了解决此问题,让我们探索两个解决方案:
1.使用泛型 (Go 1.18 )
Go 1.18 引入了泛型,允许我们创建一个通用结构来跟踪 JSON 值的存在:
type Optional[T any] struct { Defined bool Value *T }
这个可选结构使用 Defined字段来指示该值是否存在于 JSON 负载中。当该字段存在时,值字段保存未编组的值。通过将此结构合并到您自己的结构中,您可以区分 null 和缺失字段:
type Payload struct { Field1 Optional[string] `json:"field1"` Field2 Optional[bool] `json:"field2"` Field3 Optional[int32] `json:"field3"` }
2.自定义类型(预泛型)
在泛型之前,我们可以创建一个自定义类型来处理可选值:
type OptionalString struct { Defined bool Value *string }
此类型包括一个 Defined 字段来指示存在和一个用于保存未编组字符串的值字段。通过为您需要处理的每种类型定义自定义类型(例如,整数的OptionalInt),您可以保留有关字段存在的信息。
示例:
考虑以下 JSON 有效负载:
{ "somefield1":"somevalue1", "somefield2":null }
{ "somefield1":"somevalue1", }
使用我们的解决方案,我们可以区分在这些情况之间:
p1 := Payload{} _ = json.Unmarshal([]byte(`{ "somefield1": "somevalue1", "somefield2": null }`), &p1) fmt.Printf("Should be defined == true and value == nil: \n%+v\n\n", p1)
输出:
Should be defined == true and value == nil: {SomeField1:somevalue1 SomeField2:{Defined:true Value:<nil>}}
p2 := Payload{} _ = json.Unmarshal([]byte(`{ "somefield1": "somevalue1", }`), &p2) fmt.Printf("Should be defined == false: \n%+v\n\n", p2)
输出:
Should be defined == false: {SomeField1:somevalue1 SomeField2:{Defined:false Value:<nil>}}
这些示例演示了我们的解决方案如何区分空和不存在JSON字段,在反序列化过程中提供有价值的信息。
以上是如何区分 Go 中的 Null 和缺失 JSON 字段?的详细内容。更多信息请关注PHP中文网其他相关文章!