gRPC 的原理 介绍带你从头了解gRPC

news2025/1/19 2:54:01

gRPC 的原理

什么是gRPC

gRPC的官方介绍是:gRPC是一个现代的、高性能、开源的和语言无关的通用 RPC 框架,基于 HTTP2 协议设计,序列化使用PB(Protocol Buffer),PB 是一种语言无关的高性能序列化框架,基于 HTTP2+PB 保证了的高性能。

好吧我知道对于一个初学者来说这并没有意义,首先我们需要提炼关键,来回答以下几个问题,从而真正明白gRPC到底是什么?

  1. gRPC 和 RPC 的区别
  2. HTTP2 协议是什么?基于他的设计是什么意思?
  3. 序列化PB(Protocol Buffer) 是什么?

gRPC 和RPC 的区别

RPC(Remote Procedure Call)是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。

gRPC是Google开发的一个RPC系统,它使用HTTP/2协议进行通信,并使用Protocol Buffers作为接口描述语言。

这使得gRPC比传统的RPC更加强大,可以支持流式请求和响应,以及其他复杂的交互模式。

总结:RPC 是一个框架,已经写好一些功能,但是并不适合谷歌的应用场景,在这个基础上 谷歌 对RPC 框架进行了迭代开发,形成了gRPC。

HTTP2 协议是什么?基于他的设计是什么意思?

HTTP/2是HTTP协议的第二个主要版本,它在性能优化、服务器推送、头信息压缩、协议复用等方面都有显著的改进。

基于HTTP/2的设计意味着gRPC在设计时充分利用了HTTP/2的这些特性,

例如,它可以在一个TCP连接上并行发送多个请求和响应,减少了网络延迟。

总结:http2 是一个通讯规定,基于http2 的协议设计是指在使用这个通讯规定上进行的程序软件设计。

gRPC 即是基于RPC协议又基于HTTP2协议,这两种协议怎么融合的?

在明确上述两种协议的具体内容之后,我们就可以回答这个问题了。在gRPC中RPC是它的业务逻辑基本框架,HTTP2 是它的通讯逻辑基本原理

从而我们可以理解这张图:
![[Pasted image 20231114145840.png]]

二进制序列化PB(Protocol Buffer) 是什么?

Protocol Buffers(简称PB)是Google开发的一种数据序列化协议(类似于XML、JSON、hessian)。

它能够将数据结构以高效和可扩展的格式进行序列化,可以用于网络通信和数据存储等场景。

上面的解释很绕,让我们先明确什么是序列化:简单点说:就是把信息按照固定格式组织起来,为的是解决不同系统、不同编程语言、不同平台之间的传输存储问题。

序列化是指将对象转换为可以在网络传输或存储中使用的格式,以便在需要时可以重新构造出原始对象。在序列化过程中,对象的状态被转换为字节流或其他可传输的形式,以便可以在不同的系统、编程语言或平台之间进行传输或存储。

序列化的主要目的是实现对象的持久化、跨网络传输和跨平台通信。通过序列化,可以将对象转换为字节流,然后可以将字节流保存到文件中、通过网络发送给其他系统,或者在不同的程序之间进行交互。在接收方,可以通过反序列化将字节流转换回原始对象,以便进行进一步的处理或使用。

常见的序列化格式包括JSON、XML和Protocol Buffers(Protobuf)等。不同的序列化格式具有不同的特点和适用场景,选择合适的序列化格式取决于具体的需求和使用环境。

怎么用呢?这里我们以PB为例:
Protocol Buffers(简称Protobuf)使用特定的语法和编译器来定义消息的结构和字段,然后通过编译器生成相应的代码,用于序列化和反序列化对象。

下面是使用C++代码的示例,展示了如何使用Protobuf进行对象的序列化和反序列化:

  1. 定义消息结构
    首先,需要使用Protobuf的语法定义消息的结构和字段。例如,创建一个名为Person的消息类型,包含nameage字段:

    syntax = "proto3";
    
    message Person {
      string name = 1;
      int32 age = 2;
    }
    
  2. 生成代码
    使用Protobuf的编译器将上述定义的消息结构编译为相应的代码。在C++中,可以使用protoc命令行工具进行编译:

    protoc --cpp_out=. person.proto
    

    这将生成person.pb.hperson.pb.cc两个文件,包含了用于序列化和反序列化Person消息的代码。

  3. 序列化对象
    在C++代码中,可以使用生成的代码来创建Person对象,并将其序列化为字节流:

    #include "person.pb.h"
    
    // 创建Person对象
    Person person;
    person.set_name("John");
    person.set_age(25);
    
    // 序列化为字节流
    std::string serialized_data;
    person.SerializeToString(&serialized_data);
    

    SerializeToString方法将Person对象序列化为一个字符串。

  4. 反序列化对象
    反之,可以将序列化的字节流反序列化为原始的Person对象:

    // 反序列化为Person对象
    Person deserialized_person;
    deserialized_person.ParseFromString(serialized_data);
    
    // 访问反序列化后的字段
    std::string name = deserialized_person.name();
    int age = deserialized_person.age();
    

    ParseFromString方法将字节流解析为Person对象,并可以通过访问相应的字段获取数据。

需要注意的是,上述示例中的代码是使用C++语言进行的,如果使用其他编程语言,操作步骤会有所不同,但基本原理是相似的。

gRPC的工作方式

在通过上述 的详细介绍之后,我们就可以介绍gRPC 的工作方式来说明什么是gRPC 了。
![[Pasted image 20231114151128.png]]
上图中使用的是c++和红宝石以及安卓开发三种语言、平台、系统都不同的实例间通讯,展示的是gRPC 的 通用性

![[Pasted image 20231114151324.png]]
上图是解释了信息在gRPC 中是怎样流转的。展示的是gRPC 的 序列化

为什么使用gRPC

gRPC 的原理

  1. 自由,开放:让所有人,所有平台都能使用,其实就是开源,跨平台,跨语言
  2. 协议可插拔:不同的服务可能需要使用不同的消息通信类型和编码机制,例如,JSON、XML和Thirft,所以协议应允许可插拔机制,还有负载均衡,服务发现,日志,监控等都支持可插拔机制
  3. 阻塞和非阻塞:支持客户端和服务器交换的消息序列的异步和同步处理。这对于在某些平台上扩展和处理至关重要
  4. 取消和超时:一次RPC操作可能是持久并且昂贵的,应该允许客户端设置取消RPC通信和对这次通信加上一个超时时间
  5. 拒绝:必须允许服务器通过在继续处理请求的同时拒绝新请求的到来并优雅地关闭。
  6. 流处理:存储系统依靠流和流控制来表达大型数据集,其他服务,如语音到文本或股票行情,依赖于流来表示与时间相关的消息序列
  7. 流控制:计算能力和网络容量在客户端和服务器之间通常是不平衡的。流控制允许更好的缓冲区管理,以及过度活跃的对等体提供对DOS的保护。
  8. 元数据交换:认证或跟踪等常见的跨领域问题依赖于不属于服务声明接口的数据交换。依赖于他们将这些特性演进到服务,暴露API来提供能力。
  9. 标准化状态码:客户端通常以有限的方式响应API调用返回的错误。应约束状态码名称空间,以使这些错误处理决策更加清晰。如果需要更丰富的特定领域的状态,则可以使用元数据交换机制来提供该状态。
  10. 互通性:报文协议(Wire Protocol)必须遵循普通互联网基础框架

gRPC 的一些特性

  1. gRPC基于服务的思想:定义一个服务,描述这个服务的方法以及入参出参,服务器端有这个服务的具体实现,客户端保有一个存根,提供与服务端相同的服务。

  2. gRPC默认采用protocol buffer作为IDL(Interface Description Lanage)接口描述语言,服务之间通信的数据序列化和反序列化也是基于protocol buffer的,因为protocol buffer的特殊性,所以gRPC框架是跨语言的通信框架(与编程语言无关性),也就是说用Java开发的基于gRPC的服务,可以用GoLang编程语言调用

  3. gRPC同时支持同步调用和异步调用,同步RPC调用时会一直阻塞直到服务端处理完成返回结果,异步RPC是客户端调用服务端时不等待服务段处理完成返回,而是服务端处理完成后主动回调客户端告诉客户端处理完成

  4. gRPC是基于http2协议实现的,http2协议提供了很多新的特性,并且在性能上也比http1提搞了许多,所以gRPC的性能是非常好的

  5. gRPC并没有直接实现负载均衡和服务发现的功能,但是已经提供了自己的设计思路。已经为命名解析和负载均衡提供了接口

  6. 基于http2协议的特性:gRPC允许定义如下四类服务方法

    1. 一元RPC:客户端发送一次请求,等待服务端响应结构,会话结束,就像一次普通的函数调用这样简单
    2. 服务端流式RPC:客户端发起一起请求,服务端会返回一个流,客户端会从流中读取一系列消息,直到没有结果为止
    3. 客户端流式RPC:客户端提供一个数据流并写入消息发给服务端,一旦客户端发送完毕,就等待服务器读取这些消息并返回应答
    4. 双向流式RPC:客户端和服务端都一个数据流,都可以通过各自的流进行读写数据,这两个流是相互独立的,客户端和服务端都可以按其希望的任意顺序独写

gRPC的使用场景

  • 低延迟,高度可扩展的分布式系统

  • 开发与云服务器通信的客户端

  • 设计一个准确,高效,且与语言无关的新协议时

  • 分层设计,以实现扩展,例如。身份验证,负载平衡,日志记录和监控等

  • 谷歌长期以来一直在gRPC中使用很多基础技术和概念。目前正在谷歌的几个云产品和谷歌面向外部的API中使用。Square,Netflix,CoreOS,Docker,CockroachDB,Cisco,Juniper Networks以及许多其他组织和个人也在使用它。

gRPC的数据封装和数据传输问题

这里我们主要介绍pb的工作方式,即怎样封装数据;

数据封装

gRPC 使用的是自己的封装方式 Protocol Buffer简称ProtoBuf,是Google开发的一种跨语言、跨平台、可扩展的用于序列化数据协议。

下面是一个protoBuf 的样例

// XXXX.proto
// rpc服务的类 service关键字, Test服务类名
service Test {
// rpc 关键字,rpc的接口
rpc HowRpcDefine (Request) returns (Response) ; // 定义一个RPC方法
}
// message 类,c++ class
message Request {
//类型 | 字段名字| 标号
int64 user_id = 1;
string name = 2;
}
message Response {
repeated int64 ids = 1; // repeated 表示数组
Value info = 2; // 可嵌套对象
map<int, Value> values = 3; // 可输出map映射
}
message Value {
bool is_man = 1;
int age = 2;
}

以上是一个使用样例,包含方法定义、入参、出参。可以看出有几个明确的特点:

  • 有明确的类型,支持的类型有多种
  • 每个field会有名字
  • 每个field有一个数字标号,一般按顺序排列(下文编解码会用到这个点)
  • 能表达数组、map映射等类型
  • 通过嵌套message可以表达复杂的对象
  • 方法、参数的定义落到一个.proto 文件中,依赖双方需要同时持有这个文件,并依此进行编解码
    protobuf作为一个以跨语言为目标的序列化方案,protobuf能做到多种语言以同一份proto文件作为约定,不用A语言写一份,B语言写一份,各个依赖的服务将proto文件原样拷贝一份即可。

但.proto文件并不是代码,不能执行,要想直接跨语言是不行的,必须得有对应语言的中间代码才行,中间代码要有以下能力:

  • 将message转成对象,例如C++里是class,golang里是struct,需要各自表达后,才能被理解
  • 需要有进行编解码的代码,能解码内容为自己语言的对象、能将对象编码为对应的数据

在完成数据封装之后,我们就准备好了将它传输出去,接下来就是数据传输了。

数据传输

grpc采用HTTP2.0,相对于HTTP1.0 在 更快的传输 和 更低的成本 两个目标上做了改进。有以下几个基本点:

  • HTTP2 未改变HTTP的语义(如GET/POST等),只是在传输上做了优化
  • 引入帧、流的概念,在TCP连接中,可以区分出多个request/response
  • 一个域名只会有一个TCP连接,借助帧、流可以实现多路复用,降低资源消耗
  • 引入二进制编码,降低header带来的空间占用
    HTTP1.1核心问题在于:在同一个TCP连接中,没办法区分response是属于哪个请求,一旦多个请求返回的文本内容混在一起,则没法区分数据归属于哪个请求,所以请求只能一个个串行排队发送。这直接
    导致了TCP资源的闲置。
    HTTP2为了解决这个问题,提出了 流 的概念,每一次请求对应一个流,有一个唯一ID,用来区分不同的请求。基于流的概念,进一步提出了 帧 ,一个请求的数据会被分成多个帧,方便进行数据分割传输,每个帧都唯一属于某一个流ID,将帧按照流ID进行分组,即可分离出不同的请求。这样同一个TCP连接中就可以同时并发多个请求,不同请求的帧数据可穿插在一起,根据流ID分组即可。HTTP2.0基于这种二进制协议的乱序模式 (Duplexing),直接解决了HTTP1.1的核心痛点,通过这种复用TCP连接的方式,不用再同时建多个连接,提升了TCP的利用效率。

总结
传输使用 http2 协议,是在tcp链接完成之后,再在应用层发送时的一个协议,通过流 ID 和 帧报文的两个概念 解决了传输拥塞的问题。我们将这种方式称之为 tcp链接复用

gRPC 的原理

什么是gRPC

gRPC的官方介绍是:gRPC是一个现代的、高性能、开源的和语言无关的通用 RPC 框架,基于 HTTP2 协议设计,序列化使用PB(Protocol Buffer),PB 是一种语言无关的高性能序列化框架,基于 HTTP2+PB 保证了的高性能。

好吧我知道对于一个初学者来说这并没有意义,首先我们需要提炼关键,来回答以下几个问题,从而真正明白gRPC到底是什么?

  1. gRPC 和 RPC 的区别
  2. HTTP2 协议是什么?基于他的设计是什么意思?
  3. 序列化PB(Protocol Buffer) 是什么?

gRPC 和RPC 的区别

RPC(Remote Procedure Call)是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议。

gRPC是Google开发的一个RPC系统,它使用HTTP/2协议进行通信,并使用Protocol Buffers作为接口描述语言。

这使得gRPC比传统的RPC更加强大,可以支持流式请求和响应,以及其他复杂的交互模式。

总结:RPC 是一个框架,已经写好一些功能,但是并不适合谷歌的应用场景,在这个基础上 谷歌 对RPC 框架进行了迭代开发,形成了gRPC。

HTTP2 协议是什么?基于他的设计是什么意思?

HTTP/2是HTTP协议的第二个主要版本,它在性能优化、服务器推送、头信息压缩、协议复用等方面都有显著的改进。

基于HTTP/2的设计意味着gRPC在设计时充分利用了HTTP/2的这些特性,

例如,它可以在一个TCP连接上并行发送多个请求和响应,减少了网络延迟。

总结:http2 是一个通讯规定,基于http2 的协议设计是指在使用这个通讯规定上进行的程序软件设计。

gRPC 即是基于RPC协议又基于HTTP2协议,这两种协议怎么融合的?

在明确上述两种协议的具体内容之后,我们就可以回答这个问题了。在gRPC中RPC是它的业务逻辑基本框架,HTTP2 是它的通讯逻辑基本原理

从而我们可以理解这张图:
![[Pasted image 20231114145840.png]]

二进制序列化PB(Protocol Buffer) 是什么?

Protocol Buffers(简称PB)是Google开发的一种数据序列化协议(类似于XML、JSON、hessian)。

它能够将数据结构以高效和可扩展的格式进行序列化,可以用于网络通信和数据存储等场景。

上面的解释很绕,让我们先明确什么是序列化:简单点说:就是把信息按照固定格式组织起来,为的是解决不同系统、不同编程语言、不同平台之间的传输存储问题。

序列化是指将对象转换为可以在网络传输或存储中使用的格式,以便在需要时可以重新构造出原始对象。在序列化过程中,对象的状态被转换为字节流或其他可传输的形式,以便可以在不同的系统、编程语言或平台之间进行传输或存储。

序列化的主要目的是实现对象的持久化、跨网络传输和跨平台通信。通过序列化,可以将对象转换为字节流,然后可以将字节流保存到文件中、通过网络发送给其他系统,或者在不同的程序之间进行交互。在接收方,可以通过反序列化将字节流转换回原始对象,以便进行进一步的处理或使用。

常见的序列化格式包括JSON、XML和Protocol Buffers(Protobuf)等。不同的序列化格式具有不同的特点和适用场景,选择合适的序列化格式取决于具体的需求和使用环境。

怎么用呢?这里我们以PB为例:
Protocol Buffers(简称Protobuf)使用特定的语法和编译器来定义消息的结构和字段,然后通过编译器生成相应的代码,用于序列化和反序列化对象。

下面是使用C++代码的示例,展示了如何使用Protobuf进行对象的序列化和反序列化:

  1. 定义消息结构
    首先,需要使用Protobuf的语法定义消息的结构和字段。例如,创建一个名为Person的消息类型,包含nameage字段:

    syntax = "proto3";
    
    message Person {
      string name = 1;
      int32 age = 2;
    }
    
  2. 生成代码
    使用Protobuf的编译器将上述定义的消息结构编译为相应的代码。在C++中,可以使用protoc命令行工具进行编译:

    protoc --cpp_out=. person.proto
    

    这将生成person.pb.hperson.pb.cc两个文件,包含了用于序列化和反序列化Person消息的代码。

  3. 序列化对象
    在C++代码中,可以使用生成的代码来创建Person对象,并将其序列化为字节流:

    #include "person.pb.h"
    
    // 创建Person对象
    Person person;
    person.set_name("John");
    person.set_age(25);
    
    // 序列化为字节流
    std::string serialized_data;
    person.SerializeToString(&serialized_data);
    

    SerializeToString方法将Person对象序列化为一个字符串。

  4. 反序列化对象
    反之,可以将序列化的字节流反序列化为原始的Person对象:

    // 反序列化为Person对象
    Person deserialized_person;
    deserialized_person.ParseFromString(serialized_data);
    
    // 访问反序列化后的字段
    std::string name = deserialized_person.name();
    int age = deserialized_person.age();
    

    ParseFromString方法将字节流解析为Person对象,并可以通过访问相应的字段获取数据。

需要注意的是,上述示例中的代码是使用C++语言进行的,如果使用其他编程语言,操作步骤会有所不同,但基本原理是相似的。

gRPC的工作方式

在通过上述 的详细介绍之后,我们就可以介绍gRPC 的工作方式来说明什么是gRPC 了。
)
在这里插入图片描述

包浆老图

上图中使用的是c++和红宝石以及安卓开发三种语言、平台、系统都不同的实例间通讯,展示的是gRPC 的 通用性

![[Pasted image 20231114151324.png]]

上图是解释了信息在gRPC 中是怎样流转的。展示的是gRPC 的 序列化

为什么使用gRPC

gRPC 的原理

  1. 自由,开放:让所有人,所有平台都能使用,其实就是开源,跨平台,跨语言
  2. 协议可插拔:不同的服务可能需要使用不同的消息通信类型和编码机制,例如,JSON、XML和Thirft,所以协议应允许可插拔机制,还有负载均衡,服务发现,日志,监控等都支持可插拔机制
  3. 阻塞和非阻塞:支持客户端和服务器交换的消息序列的异步和同步处理。这对于在某些平台上扩展和处理至关重要
  4. 取消和超时:一次RPC操作可能是持久并且昂贵的,应该允许客户端设置取消RPC通信和对这次通信加上一个超时时间
  5. 拒绝:必须允许服务器通过在继续处理请求的同时拒绝新请求的到来并优雅地关闭。
  6. 流处理:存储系统依靠流和流控制来表达大型数据集,其他服务,如语音到文本或股票行情,依赖于流来表示与时间相关的消息序列
  7. 流控制:计算能力和网络容量在客户端和服务器之间通常是不平衡的。流控制允许更好的缓冲区管理,以及过度活跃的对等体提供对DOS的保护。
  8. 元数据交换:认证或跟踪等常见的跨领域问题依赖于不属于服务声明接口的数据交换。依赖于他们将这些特性演进到服务,暴露API来提供能力。
  9. 标准化状态码:客户端通常以有限的方式响应API调用返回的错误。应约束状态码名称空间,以使这些错误处理决策更加清晰。如果需要更丰富的特定领域的状态,则可以使用元数据交换机制来提供该状态。
  10. 互通性:报文协议(Wire Protocol)必须遵循普通互联网基础框架

gRPC 的一些特性

  1. gRPC基于服务的思想:定义一个服务,描述这个服务的方法以及入参出参,服务器端有这个服务的具体实现,客户端保有一个存根,提供与服务端相同的服务。

  2. gRPC默认采用protocol buffer作为IDL(Interface Description Lanage)接口描述语言,服务之间通信的数据序列化和反序列化也是基于protocol buffer的,因为protocol buffer的特殊性,所以gRPC框架是跨语言的通信框架(与编程语言无关性),也就是说用Java开发的基于gRPC的服务,可以用GoLang编程语言调用

  3. gRPC同时支持同步调用和异步调用,同步RPC调用时会一直阻塞直到服务端处理完成返回结果,异步RPC是客户端调用服务端时不等待服务段处理完成返回,而是服务端处理完成后主动回调客户端告诉客户端处理完成

  4. gRPC是基于http2协议实现的,http2协议提供了很多新的特性,并且在性能上也比http1提搞了许多,所以gRPC的性能是非常好的

  5. gRPC并没有直接实现负载均衡和服务发现的功能,但是已经提供了自己的设计思路。已经为命名解析和负载均衡提供了接口

  6. 基于http2协议的特性:gRPC允许定义如下四类服务方法

    1. 一元RPC:客户端发送一次请求,等待服务端响应结构,会话结束,就像一次普通的函数调用这样简单
    2. 服务端流式RPC:客户端发起一起请求,服务端会返回一个流,客户端会从流中读取一系列消息,直到没有结果为止
    3. 客户端流式RPC:客户端提供一个数据流并写入消息发给服务端,一旦客户端发送完毕,就等待服务器读取这些消息并返回应答
    4. 双向流式RPC:客户端和服务端都一个数据流,都可以通过各自的流进行读写数据,这两个流是相互独立的,客户端和服务端都可以按其希望的任意顺序独写

gRPC的使用场景

  • 低延迟,高度可扩展的分布式系统

  • 开发与云服务器通信的客户端

  • 设计一个准确,高效,且与语言无关的新协议时

  • 分层设计,以实现扩展,例如。身份验证,负载平衡,日志记录和监控等

  • 谷歌长期以来一直在gRPC中使用很多基础技术和概念。目前正在谷歌的几个云产品和谷歌面向外部的API中使用。Square,Netflix,CoreOS,Docker,CockroachDB,Cisco,Juniper Networks以及许多其他组织和个人也在使用它。

gRPC的数据封装和数据传输问题

这里我们主要介绍pb的工作方式,即怎样封装数据;

数据封装

gRPC 使用的是自己的封装方式 Protocol Buffer简称ProtoBuf,是Google开发的一种跨语言、跨平台、可扩展的用于序列化数据协议。

下面是一个protoBuf 的样例

// XXXX.proto
// rpc服务的类 service关键字, Test服务类名
service Test {
// rpc 关键字,rpc的接口
rpc HowRpcDefine (Request) returns (Response) ; // 定义一个RPC方法
}
// message 类,c++ class
message Request {
//类型 | 字段名字| 标号
int64 user_id = 1;
string name = 2;
}
message Response {
repeated int64 ids = 1; // repeated 表示数组
Value info = 2; // 可嵌套对象
map<int, Value> values = 3; // 可输出map映射
}
message Value {
bool is_man = 1;
int age = 2;
}

以上是一个使用样例,包含方法定义、入参、出参。可以看出有几个明确的特点:

  • 有明确的类型,支持的类型有多种
  • 每个field会有名字
  • 每个field有一个数字标号,一般按顺序排列(下文编解码会用到这个点)
  • 能表达数组、map映射等类型
  • 通过嵌套message可以表达复杂的对象
  • 方法、参数的定义落到一个.proto 文件中,依赖双方需要同时持有这个文件,并依此进行编解码
    protobuf作为一个以跨语言为目标的序列化方案,protobuf能做到多种语言以同一份proto文件作为约定,不用A语言写一份,B语言写一份,各个依赖的服务将proto文件原样拷贝一份即可。

但.proto文件并不是代码,不能执行,要想直接跨语言是不行的,必须得有对应语言的中间代码才行,中间代码要有以下能力:

  • 将message转成对象,例如C++里是class,golang里是struct,需要各自表达后,才能被理解
  • 需要有进行编解码的代码,能解码内容为自己语言的对象、能将对象编码为对应的数据

在完成数据封装之后,我们就准备好了将它传输出去,接下来就是数据传输了。

数据传输

grpc采用HTTP2.0,相对于HTTP1.0 在 更快的传输 和 更低的成本 两个目标上做了改进。有以下几个基本点:

  • HTTP2 未改变HTTP的语义(如GET/POST等),只是在传输上做了优化
  • 引入帧、流的概念,在TCP连接中,可以区分出多个request/response
  • 一个域名只会有一个TCP连接,借助帧、流可以实现多路复用,降低资源消耗
  • 引入二进制编码,降低header带来的空间占用
    HTTP1.1核心问题在于:在同一个TCP连接中,没办法区分response是属于哪个请求,一旦多个请求返回的文本内容混在一起,则没法区分数据归属于哪个请求,所以请求只能一个个串行排队发送。这直接
    导致了TCP资源的闲置。
    HTTP2为了解决这个问题,提出了 流 的概念,每一次请求对应一个流,有一个唯一ID,用来区分不同的请求。基于流的概念,进一步提出了 帧 ,一个请求的数据会被分成多个帧,方便进行数据分割传输,每个帧都唯一属于某一个流ID,将帧按照流ID进行分组,即可分离出不同的请求。这样同一个TCP连接中就可以同时并发多个请求,不同请求的帧数据可穿插在一起,根据流ID分组即可。HTTP2.0基于这种二进制协议的乱序模式 (Duplexing),直接解决了HTTP1.1的核心痛点,通过这种复用TCP连接的方式,不用再同时建多个连接,提升了TCP的利用效率。

总结
传输使用 http2 协议,是在tcp链接完成之后,再在应用层发送时的一个协议,通过流 ID 和 帧报文的两个概念 解决了传输拥塞的问题。我们将这种方式称之为 tcp链接复用

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1224771.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

一起Talk Android吧(第五百五十三回:解析Retrofit返回的数据)

文章目录 1. 知识回顾2. 解析方法2.1 解析有效数据2.2 解析错误数据3. 示例代码4. 经验与总结4.1 经验分享4.2 内容总结各位看官们大家好,上一回中咱们说的例子是"Retrofit的基本用法",本章回中介绍的例子是" 如何解析Retrofit返回的数据"。闲话休提,言…

解决Redis分布式锁宕机出现不可靠问题-zookeeper分布式锁

核心思想&#xff1a;当客户端要获取锁&#xff0c;则创建节点&#xff0c;使用完锁&#xff0c;则删除该节点。 客户端获取锁时&#xff0c;在 lock 节点下创建临时顺序节点。然后获取 lock下面的所有子节点&#xff0c;客户端获取到所有的子节点之后&#xff0c;如果发现自己…

hcia学习:

视频学习&#xff1a; 第一部分&#xff1a;基础学习。 19——子网掩码。 27——防火墙配置&#xff1a; 32——企业级路由器配置&#xff1a; 基础实验完成&#xff1a;&#xff08;完成以下目录对应的实验&#xff0c;第一部分基础实验就完成。&#xff09; 方法&#xff…

高校教师资格证备考

高等教育制度 关于人的全面发展和个体发展的关系&#xff0c;说法正确的是&#xff08;ABC&#xff09;。 A.个体发展是在全面发展基础上的选择性发展 B.全面发展是个体发展的前提和基础 C.个体发展又是全面发展的动力 D.个体发展是全面发展的前提和基础

Vue3-自定义hook函数

Vue3-自定义hook函数 功能&#xff1a;可以将组合式API封装成一个函数&#xff0c;用于解决代码复用的问题。注意&#xff1a;需要在src文件夹下创建一个文件夹hooks&#xff0c;在里面放js文件&#xff0c;命名随意&#xff0c;主要是将setup函数中的代码放入js文件中。 // s…

C++加持让python程序插上翅膀——利用pybind11进行c++和python联合编程示例

目录 0、前言1、安装 pybind11库c侧python侧 2、C引入bybind11vs增加相关依赖及设置cpp中添加头文件及导出模块cpp中添加numpy相关数据结构的接收和返回编译生成dll后改成导出模块同名文件的.pyd 3、python调用c4、C引入bybind11 0、前言 在当今的计算机视觉和机器学习领域&am…

x程无忧sign逆向分析

x程无忧sign逆向分析&#xff1a; 详情页sign&#xff1a; 详情页网站&#xff1a; import base64 # 解码 result base64.b64decode(aHR0cHM6Ly9qb2JzLjUxam9iLmNvbS9ndWFuZ3pob3UvMTUxODU1MTYyLmh0bWw/cz1zb3Vfc291X3NvdWxiJnQ9MF8wJnJlcT0zODQ4NGQxMzc2Zjc4MDY2M2Y1MGY2Y…

十. Linux关机重启命令与Vim编辑的使用

关机重启命令 shutdown命令 其他关机命令 其他重启命令 系统运行级别 系统默认运行级别与查询 退出登录命令logout 文本编辑器Vim Vim简介 没有菜单,只有命令Vim工作模式 Vim常用命令 插入命令 定位命令 删除命令 复制和剪切命令 替换和取消命令 搜索和搜索替换命令 保存和退出…

为什么Transformer模型中使用Layer Normalization(Layer Norm)而不是Batch Normalization(BN)

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…

使用SpringBoot Actuator监控应用

使用SpringBootActuator监控应用 微服务的特点决定了功能模块的部署是分布式的&#xff0c;大部分功能模块都是运行在不同的机器上&#xff0c;彼此通过服务调用进 行交互&#xff0c;前后台的业务流会经过很多个微服务的处理和传递&#xff0c;出现了异常如何快速定位是哪个…

【智能家居项目】FreeRTOS版本——多任务系统中使用DHT11 | 获取SNTP服务器时间 | 重新设计功能框架

&#x1f431;作者&#xff1a;一只大喵咪1201 &#x1f431;专栏&#xff1a;《智能家居项目》 &#x1f525;格言&#xff1a;你只管努力&#xff0c;剩下的交给时间&#xff01; 目录 &#x1f353;多任务系统中使用DHT11&#x1f345;关闭调度器&#x1f345;使用中断 &am…

【C语言】数组下标为啥从0开始?下标越界访问一定报错吗?

本篇文章目录 0. 相关文章1. 下标从0开始问题2. 数组下标越界不报错问题 0. 相关文章 指针与指针变量数组名不是首元素地址的的2个例外拨开指针和数组名之间的迷雾 1. 下标从0开始问题 原因是&#xff1a;数组下标访问本质是“指针解引用操作”&#xff0c;而指针又是地址&am…

gittee启动器

前言 很多小伙伴反馈不是使用gitee&#xff0c;不会寻找好的项目&#xff0c;在拿到一个项目不知道从哪里入手。 鼠鼠我呀就是宠粉&#xff0c;中嘞&#xff0c;老乡。整&#xff01;&#xff01;&#xff01; git的基本指令 在使用gitee的时候呢&#xff0c;我们只需要记住…

Java Web 实战 19 - What‘s HTTP ?

Whats HTTP ? 一 . HTTP 是什么 ?1.1 理解 HTTP 协议的工作过程1.2 HTTP 的报文格式1.2.1 准备工作1.2.2 认识 HTTP 协议的报文详情请求报文请求响应 二 . HTTP 请求报文2.1 URLURL 的 encode 2.2 HTTP 协议中的方法GETPOST常见面试题 : GET 和 POST 之间的区别 2.3 认识请求…

服务容错之限流之 Tomcat 限流 Tomcat 线程池的拒绝策略

在文章开头&#xff0c;先和大家抛出两个问题&#xff1a; 每次提到服务限流为什么都不考虑基于 Tomcat 来做呢&#xff1f;大家有遇到过 Tomcat 线程池触发了拒绝策略吗&#xff1f; JUC 线程池 在谈 Tomcat 的线程池前&#xff0c;先看一下 JUC 中线程池的执行流程&#x…

十二、Docker的简介

目录 一、介绍 Docker 主要由以下三个部分组成&#xff1a; Docker 有许多优点&#xff0c;包括&#xff1a; 二、Docker和虚拟机的差异 三、镜像和容器 四、Docker Hub 五、Docker架构 六、总结 一、介绍 Docker 是一种开源的应用容器平台&#xff0c;可以在容器内部…

一键云端,AList 整合多网盘,轻松管理文件多元共享

hello&#xff0c;我是小索奇&#xff0c;本篇教大家如何使用AList实现网盘挂载 可能还是有小伙伴不懂&#xff0c;所以简单介绍一下哈 AList 是一款强大的文件管理工具&#xff0c;为用户提供了将多种云存储服务和文件共享协议集成在一个平台上的便利性。它的独特之处在于&am…

MSYS2介绍及工具安装

0 Preface/Foreword 1 MSYS2 官网&#xff1a;MSYS2

基于Adapter用CLIP进行Few-shot Image Classification

文章目录 【ECCV 2022】《Tip-Adapter: Training-free Adaption of CLIP for Few-shot Classification》【NeuIPS 2023】《Meta-Adapter: An Online Few-shot Learner for Vision-Language Model》 【ECCV 2022】《Tip-Adapter: Training-free Adaption of CLIP for Few-shot C…

什么是缓存雪崩、击穿、穿透?

背景 数据一般是存储于数据库中&#xff0c;数据库中的数据都是存在磁盘上的&#xff0c;磁盘读写的速度相较于内存或者CPU中的寄存器来说是非常慢的了。 如果用户的请求都直接访问数据库的话&#xff0c;请求数量一上来&#xff0c;数据库很容易就崩溃了&#xff0c;所以为了…