Hi 游客

更多精彩,请登录!

比特池塘 区块链前沿 正文

我经历的 Protocol Buffers 那些坑

旧雨亲亲召
83 0 0
在我的职业生涯中,我花了很多时间来讨论protobuffers的问题。PB显然是由业余选手写的临时作品,容易陷入困境且难于编译,当然也解决了Google的问题。如果protobuffers的这些问题在序列化抽象中隔离出来,那么我也不会继续抱怨。 但不幸的是,protobuffers的糟糕设计是如此有感染力,以至于这些问题也会渗透到你的代码中。
/ h3 y' J+ U7 `) j8 w" i##由业余爱好者创建和临时性- n2 [/ B: `6 [/ d/ t1 c
我曾经在Google工作过。 Google是我第一次使用protobuffers地方(很遗憾不是最后一个)。 我今天要讨论的所有问题都存在于Google的代码库中; 这不是“错误使用protobuffers”的问题。- v% L- Z2 f. f7 P5 f
protobuffers的最大问题是其可怕的类型系统。 Java的粉丝应该感觉宾至如归,但不幸的是,大家都不认为Java有一个设计良好的类型系统。使用动态语言的人抱怨这样的类型系统太令人窒息了,而像我这样的静态语言粉丝会抱怨这种设计没有给你真正想要的类型系统。/ v8 U" A/ t0 Z2 S
临时性和业余爱好者的建立是相辅相成的。大量protobuffer规范都是亡羊补牢的做法。 规范的许多限制会让你禁不住发问,为何使用PB如此困难。然而这都只是表象,真正的原因是:6 C9 a, J% x. ?4 {5 v
Protobuffers显然是由业余爱好者建造的,因为它们为广为人知且已经解决的问题提供了不好的解决方案。4 L7 E6 ^# K( ~, P
###没有组合性
  R6 k; U. o( cProtobuffers提供了几个“特性”,但多数无法相互配合。 例如,以下是几个正交但受约束的特性列表。2 U6 x. [( v4 Y8 z
oneof字段不能repeated 。map字段具有专用语法,不能用于任何其他类型。尽管可以对map字段进行参数化,但是不支持用户定义的类型。 这意味着需要手动处理很多工作。map字段不能repeated 。map键可以是string s,但不能是bytes 。 它们也不能enum ,即使enum在protobuffer规范中实际上是整数。map值不能是其他map 。
$ M. j9 ?( d( I* F1 d: h

' g6 Z4 V( W  ]! ^- ]; a  I这种疯狂的限制列表是无原则设计和事后打补丁成的结果。 例如,oneof字段不能repeated是因为代码生成器不会产生副产品类型,而是为您提供互斥的可选字段的产品。 这种转换仅对单个字段有效。& z* o) @( {% H, z3 x6 \7 i$ M1 H
map字段无法repeated也是相关的,但揭示了类型系统的不同限制。 原理上,map应该类似于repeated Pair 。 但是因为repeated是一个语言关键词,而不是一个独立的类型,它不能再次修饰自己,因此map字段无法使用repeated。/ W$ X* u9 d: e1 {7 w- u7 z0 W" U' U
估计你对为什么enum不能用作map键,已经有了自己猜测。(译者注:没想到的,应该参加GIAC学习了)
1 u* y) {5 h# \& X! v+ [令人沮丧的是,哪怕对现代类型系统如何工作有一点理解,就可以在大大简化 protobuffer规范的同时消除很多限制。
* q, G- J7 Z/ l' {. A% J1 R6 n解决方案如下:
- F0 D/ |* E) E  h% }/ E* v# Y使所有字段都必须是required。这使得消息都是产品类型。将oneof字段替换为独立数据类型。这提供副产品类型。提供通过其他类型参数化产品和副产品类型的能力。, c1 I5 F) |* d. z
  F  w7 ~1 w& x! k
只需要这三个功能,你就能定义任何可能的数据类型。 我们可以根据它们重新实现其余的protobuffer规范。1 I; z) O/ C8 J/ y7 ~- t! @
例如,我们可以重建optional字段:2 x  x5 M. B( D
( l. M7 e( K1 G% V) P% I9 O! i; B
构建repeated字段也很简单:  b' \* O9 H5 H7 u
/ x$ X" I1 H) R, Y8 V% m
当然,实际允许的序列化逻辑比通过网络推送链表更高明 - 毕竟, 实现和语义不需要一对一对齐8 O! r$ v1 x; ?, \: d
###存疑的选择
9 V/ F, {4 v' f. I在Java的基础上,protobuffers区分了标量类型和消息类型。 标量类型或多或少与机器原语相对应 - 比如int32 , bool和string 。 另一方面,其他类型都是消息类型。 所有库和用户定义的类型都是消息类型。
2 \  {! U6 _3 Q4 }2 L5 d7 m当然,这两种类型的语义完全不同。. v, ?; A- A) @5 `3 l  Y
即使你没有设置它们,标量类型的字段也存在。 我提到过(至少在proto 3中 )所有protobuffers都可以零初始化。标量字段获取false-y值—例如, uint32初始化为0 , string初始化为"" 。将protobuffer中缺失的字段与默认值字段区分开来是不可能的。 这么做是为了优化默认值(减少传输的数据)。0 O" E$ l  U1 h1 h9 y
protobuffers声称可以向后向和向前兼容,然而无法区分未设置值和默认值是一场噩梦。 如果确实是为了每个字段节约一位(有或没有)而做出如此设计,那么有点不值当。. o: a5 ]! r/ }) y. W
相比之下, 虽然标量类型设计的不够好,但消息类型字段的行为就完全放飞自我了。消息类型字段无论是否存在,它们的行为都异常疯狂。 其访问代码值得细细剖析。 假设如下伪Java代码:
* O+ ^8 t9 o3 |* a, @/ e! n3 M( Z& I1 D8 C
我们的想法是,如果未设置foo字段,则无论何时请求都会看到默认初始化的副本,但实际上不会修改其容器。 但是如果修改foo ,它也会修改它的父级! 所有这一切只是为了避免使用Maybe Foo类型和相关的细微差别,需要弄清楚未设置值意味着什么。0 B# p$ s6 ~* r
这种行为特别令人震惊,因为它破坏了规律! 我们期望赋值不会引起别的动作。 而PB将悄悄地更改msg以获得foo的零初始化副本。
- K! n1 I% O2 @  _0 L与标量字段不同,我们至少可以检测消息字段是否未设置。 protobuffers提供了生成的bool has_foo()方法。如果想复制foo,则需要编写以下代码:
) O# B; {9 O1 n+ c, ^' ^( V" L
, W$ a$ `% @0 b" v' d7 r/ }. l( N请注意,至少在静态类型语言中,由于方法foo() , set_foo()和has_foo()之间的命名关系, 我们无法抽象处理。 除了预处理器宏之外,我们无法以编程方式生成它们:
# R: }. H2 g* j2 l# r! h
1 C  K$ @1 _6 @9 a. e0 X(但预处理器宏是由Google code style指南禁止的。)
, c( M- P. Q7 z# i4 z+ x& _9 `如果所有可选字段都被实现为Maybe s,那么将很容易抽象处理这种情况。
, u4 D) X) J3 ]7 m+ z& ~让我们谈谈另一个有问题的决定。 虽然你可以在protobuffers中定义一个字段,但它们的语义不是副产品类型! 相反,对于每种情况你得到一个可选字段,以及setter中的魔术代码。如果设置了一个,它会将其他情况清除掉。1 H% i' f6 g" M  q. i+ ]% C
乍一看,这似乎应该在语义上等同于union类型。 但相反,它是bug之源! 这种行为允许默默地删除任意数量的数据! 在protobuffers上编写通用的,无错误的,多态的代码实际上是不可能的。. Z+ A- p6 `# g" Q( E9 ~6 X  g
这不是任何人都喜欢听到的东西,更不用说我们这些已经爱上参数多态性的人 - 这给了我们完全相反的承诺。
% a0 ^, t+ p9 ^###向后兼容的谎言  r' A; w- X  V# u! P+ @
protobuffers的另一个杀手特性是它们“编写向后兼容API的能力”。; ^% Q9 F3 }" G/ g# X. i' m% f" v
protobuffers 默认情况下通过偷偷地执行错误操作来实现其兼容性。 当然,谨慎的程序员可以(并且应该)会对接收到的protobuffers消息进行检查。 但是你需要不停编写防御性检查代码以确保您的数据没有问题,也许这只是意味着反序列化步骤过于宽松。 您所能做的就是将健全性检查逻辑从定义良好的边界中分散开来,并将扩散整个代码库中。' S( S% f2 s( J/ s% c1 d
另一个论点是,protobuffers将保留他们不理解的消息中存在的任何信息。 原则上,这意味着发送路由消息(不知道其schema版本)是非破坏性的。
0 c+ a+ J7 q8 ~当然,在纸面上它是一个很酷的功能。 但我从来没有见过一个真正保留该属性的应用程序。 除了路由软件之外,没有什么其他软件仅检查消息的某些位然后在未更改的情况下转发消息。 使用protobuffers的绝大多数程序将解码消息,将其转换为别的消息,并将其发送给其他程序。 这些变换是需要手动编码的。 从一个protobuffer到另一个protobuffer的手动编码转换不会保留两者之间的未知字段,因为它实际上毫无意义。
. p7 k6 H6 c% W* s% P& t% y/ `9 ?这种对待protobuffers的态度总是与其他丑陋的方式并举。protobuffers的风格指南积极倡导反DRY,并建议尽可能内联定义。 这背后的原因是,如果这些定义在将来发生分歧,它允许您单独修改消息。- g" ]7 J$ N7 w  l# G* T
这个问题的根源在于Google将数据的含义与其物理表示混为一谈。 当你处于谷歌规模时,这种事情可能是有道理的。 毕竟,他们有一个内部工具,允许您比较程序员时间与网络利用率背后(或者其他事情)的成本。 与大多数公司不同,工程师薪水是谷歌最小的开支之一。 从财务角度来说,浪费程序员的时间以减少几个字节是有道理的。
9 ?0 h3 A9 F; I在排名前五的科技公司之外,我们都跟Google的规模差了好几个数量级。 你的创业公司不应该浪费工程师的时间来削减字节数。 但是削减字节并浪费程序员的时间正是protobuffers优化的原因。  \: v9 e$ M# L; d/ D, X9 {) u" A7 K
面对现实吧。大部分公司永远达不到Google的规模。 对那些只是因为“谷歌使用它”,因此“它是行业最佳实践”的技术,我们应该停止搬到自己公司。
- U$ Q* }: u' u5 C###Protobuffers污染代码库
' h; {5 D$ }4 c& d8 T/ E2 I如果可以将protobuffer的使用限制在网络传输,我就不会如此为难。 不幸的是,虽然原则上有一些解决方案,但它们都不足以实际用于真实软件。4 ^' C. k; k$ Z5 r) Y' c- \
Protobuffers对应于您希望发送的数据,这通常与应用程序要使用的实际数据相关但不相同 。 这使我们处于一种令人不安的境地,需要在三种不良选择中选择一种:9 i2 P. |' g( z6 I+ D2 l' u
维护一个描述您实际需要的数据的单独类型,并确保两者同步。将数据打包成传输格式以供应用程序使用。每次需要时都可以通过传输格式获取信息。
6 x+ x+ i) L% O  M

4 a" I, z7 T$ H% X0 D选项1显然是“正确的”解决方案,但它与protobuffers无法匹配。 该语言的功能不够强大,无法同时作为传输格式和应用程序数据格式。这意味着需要写一个完全独立的数据类型,与protobuffer同步,并在两者之间显式使用序列化代码来同步。而大部分人使用PB就是为了不写序列化代码,所以这种情况不会发生。" {% V" ?7 n( s( c& a- T
相反,使用protobuffers的代码会在整个代码库中扩散。 我在谷歌的参与的主要项目是一个编译器,它用各种各样的protobuffer作为输入,并在另一个程序中输出一个等价的“程序”。 输入和输出格式表达能力都足够,然而保持适当并行的C++版本永远不工作。代码无法利用我们为编写编译器而实现的任何丰富技术,因为protobuffer(以及由此产生的代码)过于僵化,无法做任何有趣的事情。; R4 l: b" C1 L: H+ Q
结果是,可能有50行递归代码就能实现的事情需要10,000行PB代码。 我想实现的功能因为PB的限制而无法实现。
7 N: m7 d' S6 J4 }+ J2 N虽然这是仅仅是一个例子,但它不是孤立的。 由于它们严格的代码生成,语言中的protobuffers的表现形式从来都不是惯用的方式。
7 v6 s, n5 b( D8 [7 t8 {但即使这样,你仍然需要将一个糟糕的类型系统嵌入到目标语言中。 因为大多数protobuffers的功能都是不完善的,这些令人讨厌的属性也会泄漏到我们的代码库中。 这意味着我们不仅要实现,而且还要在任何与之交互的项目中延续这些糟糕的想法。
* C' g, u% r  \- R0 S在坚实的基础上实现无用的功能很容易,但走向反面则是是挑战。! ^4 d' v$ M% V( I8 U# ?
简而言之,放弃将protobuffers引入项目吧。) O: @2 I( y* ^8 H$ X  h9 r
BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

成为第一个吐槽的人

旧雨亲亲召 小学生
  • 粉丝

    0

  • 关注

    0

  • 主题

    6