proto2 proto3 变化

总的来说,proto3 比 proto2 支持更多语言但 更简洁。去掉了一些复杂的语法和特性,更强调约定而弱化语法。如果是首次使用 Protobuf ,建议使用 proto3 。

1、在第一行非空白非注释行,必须写:

syntax = “proto3”, 否则默认版本是proto2;

2、字段规则移除了 “required”,并把 “optional” 改名为 “singular”;

在 proto2 中 required 也是不推荐使用的。proto3 直接从语法层面上移除了 required 规则。

3、“repeated”字段默认采用 packed 编码;

在 proto2 中,需要明确使用 [packed=true] 来为字段指定比较紧凑的 packed 编码方式。

4、语言增加 Go、Ruby、JavaNano 支持;

5、移除了 default 选项;

在 proto2 中,可以使用 default 选项为某一字段指定默认值。在 proto3 中,字段的默认值只能根据字段类型由系统决定。也就是说,默认值全部是约定好的,而不再提供指定默认值的语法。

在字段被设置为默认值的时候,该字段不会被序列化。这样可以节省空间,提高效率。

但这样就无法区分某字段是根本没赋值,还是赋值了默认值。这在 proto3 中问题不大,但在 proto2 中会有问题。

比如,在更新协议的时候使用 default 选项为某个字段指定了一个与原来不同的默认值,旧代码获取到的该字段的值会与新代码不一样。

  • string, 默认值是空字符串(empty string)
  • bytes, 默认值是空bytes(empty bytes)
  • bool, 默认值是false
  • numeric, 默认值是0
  • enum, 默认值是第一个枚举值(value必须为0)
  • repeated,默认值为empty,通常是一个空list

6、枚举类型的第一个字段必须为 0 ;

这也是一个约定。

7、移除了对分组的支持;

分组的功能完全可以用消息嵌套的方式来实现,并且更清晰。在 proto2 中已经把分组语法标注为『过期』了。这次也算清理垃圾了。

8、proto3代码在解析新增字段时,会把不认识的字段丢弃,再序列化后新增的字段就没了;

在 proto2 中,旧代码虽然会忽视不认识的新增字段,但并不会将其丢弃,再序列化的时候那些字段会被原样保留。

我觉得还是 proto2 的处理方式更好一些。能尽量保持兼容性和扩展能力,或许实现起来也更简单。proto3 现在的处理方式,没有带来明显的好处,但丢掉了部分兼容性和灵活性。

[2017-06-15 更新]:经过漫长的讨论,官方终于同意在 proto3 中恢复 proto2 的处理方式了。

9、移除了对扩展的支持,新增了 Any 类型;

Any 类型是用来替代 proto2 中的扩展的。目前还在开发中。

proto2 中的扩展特性很像 Swift 语言中的扩展。理解起来有点困难,使用起来更是会带来不少混乱。

相比之下,proto3 中新增的 Any 类型有点像 C/C++ 中的 void* ,好理解,使用起来逻辑也更清晰。

10、增加了 JSON 映射特性;

你可能感兴趣的:(protobuffer,Protocol,Buffers)