建筑网站官网,南京家装公司有哪些品牌,python可以做网站,营销推广型网站来自公众号#xff1a;新世界杂货铺这两天翻了翻以前的项目#xff0c;发现不同项目中关于Protobuf 3缺失值和默认值的区分居然有好几种实现。今天笔者冷饭新炒#xff0c;结合项目中的实现以及切身经验共总结出如下六种方案。增加标识字段众所周知#xff0c;在Go中数字类…来自公众号新世界杂货铺这两天翻了翻以前的项目发现不同项目中关于Protobuf 3缺失值和默认值的区分居然有好几种实现。今天笔者冷饭新炒结合项目中的实现以及切身经验共总结出如下六种方案。增加标识字段众所周知在Go中数字类型的默认值为0这里仅以数字类型举例这在某些场景下往往会引起一定的歧义。以is_show字段为例如果没有该字段表示不更新DB中的数据如果有该字段且值为0则表示更新DB中的数据为不可见如果有该字段且值为1则表示更新DB中的数据为可见。上述场景中实际要解决的问题是如何区分默认值和缺失字段。增加标识字段是通过额外增加一个字段来达到区分的目的。例如增加一个has_show_field字段标识is_show是否为有效值。如果has_show_field为true则is_show为有效值否则认为is_show未设置值。此方案虽然直白但每次设置is_show的值时还需设置has_show_field的值甚是麻烦故笔者十分不推荐。字段含义和默认值区分字段含义和默认值区分即不使用对应类型的默认值作为该字段的有效值。接着前面的例子继续描述is_show为1时表示展示is_show为2时表示不展示其他情况则认为is_show未设置值。此方案笔者还是比较认可的唯一问题就是和开发者的默认习惯略微不符。使用oneofoneof 的用意是达到 C 语言 union 数据类型的效果但是诸多大佬还是发现它可以标识缺失字段。message Status {oneof show {int32 is_show 1;}
}
message Test {int32 bar 1;Status st 2;
}
上述proto文件生成对应go文件后Test.St为Status的指针类型故通过此方案可以区分默认值和缺失字段。但是笔者认为此方案做json序列化时十分不友好下面是笔者的例子// oneof to json
ot1 : oneof.Test{Bar: 1,St: oneof.Status{Show: oneof.Status_IsShow{IsShow: 1,},},
}
bts, err : json.Marshal(ot1)
fmt.Println(string(bts), err)
// json to oneof failed
jsonStr : {bar:1,st:{Show:{is_show:1}}}
var ot2 oneof.Test
fmt.Println(json.Unmarshal([]byte(jsonStr), ot2))
上述输出结果如下{bar:1,st:{Show:{is_show:1}}} nil
json: cannot unmarshal object into Go struct field Status.st.Show of type oneof.isStatus_Show通过上述输出知oneof的json.Marshal输出结果会额外多一层而json.Unmarshal还会失败因此使用oneof时需谨慎。使用wrapper类型这应该是google官方提出的解决方案我们看看下面的例子import google/protobuf/wrappers.proto;
message Status {google.protobuf.Int32Value is_show 1;
}
message Test {int32 bar 1;Status st 2;
}
使用此方案需要引入google/protobuf/wrappers.proto。此方案生成对应go文件后Test.St也是Status的指针类型。同样我们也看一下它的json序列化效果wra1 : wrapper.Test{Bar: 1,St: wrapper.Status{IsShow: wrapperspb.Int32(1),},
}
bts, err json.Marshal(wra1)
fmt.Println(string(bts), err)
jsonStr {bar:1,st:{is_show:{value:1}}}
// 可正常转json
var wra2 wrapper.Test
fmt.Println(json.Unmarshal([]byte(jsonStr), wra2))
上述输出结果如下{bar:1,st:{is_show:{value:1}}} nil
nil和oneof方案相比wrapper方案的json反序列化是没问题的但是json.Marshal的输出结果也会额外多一层。另外经笔者在本地试验此方案无法和gogoproto一起使用。允许proto3使用optional标签前面几个方案估计在实践中还是不够尽善尽美。于是2020年5月16日protoc v3.12.0发布该编译器允许proto3的字段也可使用 optional修饰。下面看看例子message Status {optional int32 is_show 1;
}
message Test {int32 bar 1;Status st 2;
}
此方案需要使用新版本的protoc且必须使用--experimental_allow_proto3_optional开启此特性。protoc升级教程见https://github.com/protocolbuffers/protobuf#protocol-compiler-installation。下面继续看看该方案的json序列化效果var isShow int32 1
p3o1 : p3optional.Test{Bar: 1,St: p3optional.Status{IsShow: isShow},
}
bts, err json.Marshal(p3o1)
fmt.Println(string(bts), err)
var p3o2 p3optional.Test
jsonStr {bar:1,st:{is_show:1}}
fmt.Println(json.Unmarshal([]byte(jsonStr), p3o2))
上述输出结果如下{bar:1,st:{is_show:1}} nil
nil据上述结果知此方案与oneof以及wrapper方案的json序列化相比更加符合预期同样经笔者在本地试验此方案无法和gogoproto一起使用。proto2和proto3结合使用作为一个gogoproto的忠实用户笔者希望在能区分默认值和缺失值的同时还可以继续使用gogoproto的特性。于是便产生了proto2和proto3结合使用的野路子。// proto2
message Status {optional int32 is_show 2;
}
// proto3
message Test {int32 bar 1 [(gogoproto.moretags) form:more_bar, (gogoproto.jsontag) custom_tag];p3p2.Status st 2;
}
需要区分缺失字段和默认值的message定义在语法为proto2的文件中proto3通过import导入proto2的message以达区分目的。optional修饰的字段在Go中会生成指针类型因此区分缺失值和默认值就变的十分容易了。下面看看此方案的json序列化效果// p3p2 to json
p3p21 : p3p2.Test{Bar: 1,St: p3p2.Status{IsShow: isShow},
}
bts, err json.Marshal(p3p21)
fmt.Println(string(bts), err)
var p3p22 p3p2.Test
jsonStr {custom_tag:1,st:{is_show:1}}
fmt.Println(json.Unmarshal([]byte(jsonStr), p3p22))
上述输出结果如下{custom_tag:1,st:{is_show:1}} nil
nil根据上述结果知此方案不仅能够活用gogoproto的各种tag其结果也和在proto3中直接使用optional效果一致。虽然笔者已经在自己的项目中使用了此方案但是仍然要提醒一句“写本篇文章时笔者特意去github看了gogoproto的发布日志gogoproto最新一个版本发布时间为2019年10月14日笔者大胆预言gogoproto以后不会再更新了所以此方案还请大家酌情使用”。最后衷心希望本文能够对各位读者有一定的帮助。注 1. 文中笔者所用go版本为go1.15.2 2. 文中笔者所用protoc版本为3.14.0 3. 文章中所用完整例子https://github.com/Isites/go-coder/blob/master/pbjson/main.go