找回密码
 立即注册
查看: 501|回复: 0

如果面试提到Protobuf,面试官问其道理怎么办?

[复制链接]
发表于 2023-9-3 07:42 | 显示全部楼层 |阅读模式
计算机网络编程中一个非常基本的问题:该怎样暗示client与server之间交互的数据,在往下看之前先想一想这个问题。
共识与协议

这个问题可不像看上去的那样简单,因为client进程和server进程运行在分歧的机器上,这些机器可能运行在分歧的措置器平台、可能运行在分歧的操作系统、可能是由分歧的编程语言编写的,server要怎样才能识别出client发送的是什么数据呢?就像这样:


client给server发送了一段数据:
0101000100100001server怎么能知道该怎样“解读”这段数据呢?
显然,client和server在发送数据之前必需首先达成某种关于怎样解读数据的共识,这就是所谓的协议
这里的协议可以是这样的:“将每8个比特为一个单元解释为无符号数字”,如果协议是这样的,那么server接收到这串二进制后就会将其解析为81(01010001)与33(00100001)。
当然,这里的协议也可以是这样的:“将每8个比特为一个单元解释为ASCII字符”,那么server接收到这串二进制后就将其解析为“Q!”。
可见,同样一串二进制在分歧的“上下文/协议”下有完全纷歧样的解读,这也是为什么计算机明明只认知0和1但是却能措置非常复杂任务的底子原因,因为一切都可以编码为0和1,同样的我们也可以从0和1中解析出我们想要的信息,这就是所谓的编解码技术。
实际上不止0和1,我们也可以将信息编码为摩斯暗码(Morse code)等,只不外计算机擅长措置0和1而已。


扯远了,回到本文的主题。
长途过程调用:RPC

作为法式员我们知道,client以及server之间不会简单传递一串数字以及字符这样简单,尤其在互联网大厂后端处事这种场景下。
当我们在电商App搜索商品、打车App呼叫出租车以及刷短视频时,每一次请求的背后在后端都涉及大量处事之间的交互,就像这样:


完成一次客户端请求gateway这个处事要调用N多个下游处事,所谓调用是说A处事向B处事发送一段数据(请求),B处事接收到这段数据后执行相应的函数,并将成果返回给A处事。
只不外对于处事A来说并不想关心网络传输这样的底层细节,如果能像调用当地函数一样调用长途处事就好了,这就是所谓的RPC,经典的实现方式是这样的:


RPC对上层提供和普通函数一样的接口,只不外在实现上封装了底层复杂的网络通信,RPC框架是当前互联网后端的基石之一,很多所谓互联网后端的职位无非就是在此基础之上堆业务逻辑。
本文我们不关心此中的细节,这里我们只关心在网络层client是怎样对请求参数进行编码、server怎样对请求参数进行解码的,也就是本文开头提出的问题。
相关视频保举
protobuf序列化协议工程应用方式和实践分析
高性能处事器通信协议设计之xml-json-protobuf对比分析
C++后端必读7个开源项目源码(redis、mysql、nginx、protobuf、libevent、cjson、log4cpp)
需要C/C++ Linux处事器架构师学习资料加qun579733396获取(资料包罗C/C++,Linux,golang技术,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK,ffmpeg等),免费分享


信息的编解码

在思考怎样进行编解码之前我们必需意识到:

  • client和server可能是用分歧语言编写的,你的编解码方案必需通用且不能和语言绑定
  • 编解码方式的性能问题,尤其是对时间要求苛刻的处事
首先,我们最应该能想到的就是以纯文本的形式来暗示。
纯文本从来都是一种非常有友好的信息载体,为什么?很简单,因为人类(我们)可以直接看懂,就像这段:
{
”widget”: {
  ”window”: {
   ”title”: ”Sample Konfabulator Widget”,
   ”name”: ”main_window”,
   ”width”: 500,
   ”height”: 500
  },
  ”image”: {
   ”src”: ”Images/Sun.png”,
   ”name”: ”sun1”,
   ”hOffset”: 250,
   ”vOffset”: 250,
  },
}
}是不是很清晰,一目了然,只要我们实现约定好文本的布局(也就是语法),那么client和server就能操作这种文本进行信息的编码以及解码,不管client和server是运行在x86还是Arm、是32位的还是64位的、运行在Linux上还是windows上、是大端还是小端,都可以无障碍交流。
因此在这里,文本的语法就是一种协议。


趁便说一句,你都规定好了文本的语法,实际上就相当于发现了一种语言
这里用来举例用的语言就是所谓的Json,只不外json这种语言不是用来暗示逻辑(代码)而是用来存储数据的。
Json就是这个老头提出来的:


除了Json,另一种操作文本存储数据的暗示方式是XML,来一段感到感染下:
<note>
<to>Tove</to>
<from>Jani</from>
<heading>Reminder</heading>
<body>Don&#39;t forget me this weekend!</body>
</note>相对Json来说是不是就没那么容易看懂了,Json呈现后在web范围逐渐代替了XML。
当两段数据量很少的时候——就像浏览器和处事端的交互,Json可以工作的非常好,这个场景就是这里:


在这里是json的天下。
但对于后端处事之间的交互来说就纷歧样了,后端处事之间的RPC调用可能会传输大量数据,如果全部用纯文本的形式来暗示数据那么不管是网络带宽还是性能可能城市差强人意。


在这种场景下,Json并不是最好的选项,主要原因之一就在于性能以及数据的体积。
我们知道,文本暗示对人类是最友好的,对机器来说则不是这样,对机器来说最好的还是01二进制。
那么有没有二进制的编码方式吗?答案是必定的,这就是当前互联网后端中风行的protobuf,Google公司开源项目。
那么protobuf有什么神奇之处吗?
假设client端想给server端传输这样一段信息:“我有一个id,其值为43”,那么在XML下是这样暗示的:
<id>43</id>数一数这这段数据占据了多少字节,很显然是11字节;
而如果用json来暗示呢?
{”id”:43}数一数这段数据占据了多少字节,显然是9字节;
而如果用protobuf来暗示呢? 是这样的:
// 动静定义
message Msg {
  optional int32 id = 1;
}

// 实例化
Msg msg;
msg.set_id(43);此中Msg的定义看上去比Json和XML更加复杂了,但这些只是给人看的,这些还会被protbuf进一步措置,最终被编码为:
082b也就是0x08与0x2b,这占据了多少字节呢?答案是2字节。
从json的9字节到protobuf的2字节,数据大小减少了4倍多,数据量的减少意味着:

  • 更少的网络带宽
  • 更快的解析速度
那么protobuf是怎样做到这一点的呢?
protobuf是怎样实现的?

首先,我们来思考最简单的情况,该怎样暗示数字。
你可能会想这还不简单,统一用固定长度,比如用64个比特(8字节),这种方式可行,但问题是非论一个数字有多小,比方2,那么用这种方式暗示2也需要占据64个比特(8字节):


明明只要一个字节就能暗示而我们却用了8个,前面的全都是0,这也太奢侈太浪费了吧。
显然,在这里我们不能使用固定长度来暗示数字,而需要使用变长方式来暗示。
什么叫变长?意思是说如果数字本身斗劲大,那么其使用的比特位可以较多,但如果数字很小那么就应该使用较少的比特位来暗示,这就叫变长,随机应变,不死板。
那怎样变长呢?
我们规定:对于每一个字节来说,第一个比特位如果是1那么暗示接下来的一个比特依然要用来解释为一个数字,如果第一个比特为0,那么说明接下来的一个字节不是用来暗示该数字的。
也就是说对于每个8个比特(1字节)来说,它的有效载荷是7个比特,第一个比特仅仅用来标识表记标帜是否还应该把接下来的一个字节解析为数字。
按照这个规定假设来了这样一串01二进制:
1010110000000010按照规定,我们首先取出第一个字节,也就是:
10101100此时我们发现第一个比特位是1,因此我们知道接下来的一个字节也属于该数字,将当前字节的1去掉就是:
0101100然后我们看下一个字节:
00000010我们发现第一个bit为0,因此我们知道下一个字节不属于该数字了。
接下来我们将解析到的0101100(第一个字节去掉第一个比特位)以及第二个字节0000010(第二个字节去掉第一个比特位)翻转之后拼接到一起,这里之所以翻转是因为我们规定数字的高位在后。
这个过程就是:
    1010110000000010  
->  10101100 | 00000010 // 解析得到两个字节
    _          _

->  0101100  |  0000010  // 各自去掉最高位
->  0000010  |  0101100  // 两个字节翻转挨次

    0000010  +  0101100
->  100101100           // 拼接 最后我们得到了100101100,这一串二进制暗示数字300。
这种数字的变长暗示方式在protobuf中被称之为varint。
因此在这种暗示方式下,如果数字较大,那么使用的比特就多,如果数字较小那么使用比特就少,聪明吧。
有的同学看到这里可能会问题,刚才讲解的方式只能暗示无符号数字,那么有符号数字该怎么暗示呢?比如-2该怎么暗示?
有符号数的暗示

按照刚才变长编码的思想,-2147483646使用的比特位应该比-2要少。
然而我们知道在计算机世界中负数使用补码暗示的,也就是说最高位(最左侧的比特位)必然是1,假设我们使用64位来暗示数字,那么如果我们依然用补码来暗示数字的话那么无论这个负数有多大还是多小都需要占据10个字节的空间。
为什么是10个字节呢?
不要忘了varint每个字节的有效负荷是7个比特,那么对于需要64位暗示的数字来说就需要64/7向上取整也就是10个字节来暗示。
这显然不能满足我们对数字变长存储的要求。
该怎么解决这个问题呢?
既然无符号数字可以便利的进行变长编码,那么我们将有符号数字映射称为无符号数字不就可以了,这就是所谓的ZigZag编码,是不是很聪明,就像这样:
原始信息      编码后
0            0
-1           1
1            2
-2           3
2            4
-3           5
3            6

...          ...

2147483647   4294967294
-2147483648  4294967295这样我们就可以将有符号数字转为无符号数字,接收方接收到该数据后再恢复出有符号数字。
此刻数字的问题彻底解决了,但这仅仅是万里长征第一步。
字段名称与字段类型

对于任何一个有用的信息都包含这样几部门:

  • 字段名称
  • 字段类型
  • 字段值
就像C/C++中定义变量时:
int i = 100;在这里,字段名称就是i,字段类型是int,字段值是100。
刚才我们用varint以及ZigZag编码解决了字段值暗示的问题,那么该怎样暗示字段名称和字段类型呢?
首先,对于字段类型还斗劲简单,因为字段类型就那么多,protobuf中定义了6种字段类型:


对于6种字段类型我们使用3个比特位来暗示就足够了。
接下来斗劲有趣的是字段名称该怎么暗示呢?假设我们需要传递这样一个字段:
int long_long_name = 100;那么我们真的需要把“long_long_name”这么多字符通过网络传递给对端吗?
既然通信双方需要协议,那么“long_long_name”这字段其实是client和server都知道的,它们独一不知道的就是“哪些值属于哪些字段”。
为解决这个问题,我们给每个字段都进行编号,比如通信双方都知道“long_long_name”这个字段的编号是2,那么对于:
int long_long_name = 100;这个信息我们只需要传递:

  • 字段名称:2 (2对应字段“long_long_name”)
  • 字段类型:0 (0暗示varint类型,参见上图)
  • 字段值:100
所以我们可以看到,无论你用多么复杂的字段名称也不会影响编码后占据的空间,字段名称底子就不会呈此刻编码后的信息中,so clever。
从宏不雅观上看

我们已经在protobuf中看到了数字以及字段名称以及字段类型是怎么暗示了,此刻是时候从宏不雅观角度来看看多个字段该怎么编码了。
从本质上讲,protobuf被编码后形成一系列的key-value,每个key-value对应一个proto中的字段。
也就是键值对:


此中value斗劲简单,也就是字段值;而字段名称和字段类型会被拼接成key,protobuf中共有6种类型,因此只需要3个比特位即可;字段名称只需要存储对应的编号,这样可以就可以这样编码:
(字段编号 << 3) | 字段类型假设server接收到了一个key为0x08,其二进制的暗示为:
0000 1000由于key也是操作varint编码的,因此需要将第一个比特位去掉,这样我的得到:
000 1000按照key的编码方式,其后三个比特位暗示字段类型,即:
000也就是0,这样我们知道该key的类型是Varint(第0号类型),而字段编号为抹掉后3个比特位的值,即:
0001这样,我们就知道了该key对应的字段编号为1,得到编号我们就能按照编号找到对应的编号名称。
嵌套数据

与Json和XML类似,protobuf中也撑持嵌套动静,就像这样:
message SubMsg {
  optional int32 id = 1;
}其实现也斗劲简单,这依然遵循被编码后形成一系列的key-value,只不外对于嵌套类型的key来说,其value是由子动静的key-value组成。


protobuf与编译语言

与Json一样,protobuf也是一门语言,兼具了文本的可读性以及二进制的高效。
protobuf之所以能做到这一点就比如C语言与机器指令。
C语言是给法式员看的,可读性好,而机器指令是给硬件使用的,性能好,编译器会将C语言法式转为机器可执行的机器指令。
而protobuf也一样,protobuf也是一门语言,会将可读性较好的动静编码为二进制从而可以在网络中进行传布,而对端也可以将其解码回来。
在这里protobuf中定义的动静就比如C语言,编码后的二进制动静就比如机器指令。
而protobuf作为事实上语言必然有本身的语法,其语法就是这样:


怎么样,还感觉编译道理没什么用吗?
不理解编译道理是不成能发现protobuf这种技术的。
总结

我在写这篇文章时不竭感慨,Google的这项技术节省了多少法式员的时间,同时我们也能看到这种基石般的技术依赖的底层道理却非常古老:

  • 信息的编解码
  • 编译道理
怎么样,这些是不是远远没有IT界各种风行的技术听上去时髦有趣,而正是这种朴素的技术支撑起了工业界,此刻你也应该能大白底层技术的重要性了吧。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

×
懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Unity开发者联盟 ( 粤ICP备20003399号 )

GMT+8, 2025-1-9 01:36 , Processed in 0.105610 second(s), 28 queries .

Powered by Discuz! X3.5 Licensed

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表