RPC是什么?和HTTP区别?

news2025/1/25 0:04:28

RPC 是什么?HTTP 是什么?

作为一个程序员,假设我们需要从A电脑的进程发送一段数据到B电脑的进程,我们一般会在代码中使用 Socket 进行编程。

在这里插入图片描述

此时,可选性一般就是 TCP 和 UDP 二选一,由于 TCP 可靠、UDP 不可靠,只要对可靠性有要求,一般就选 TCP 了。

Socket 本质是个代码库,我们需要先创建类似于:

在这里插入图片描述

其中 SOCK_STREAM是指使用字节流传输数据。说白了就是 TCP 协议。

在这里插入图片描述

在定义了 socket 之后,我们就可以对这个 socket 进行操作。
比如使用 bind() 方法绑定 IP 端口;
connect() 方法发起建联,里面就包含了常见的 TCP 三次握手流程;
在建立连接之后就可以使用 send() 方法发送数据;
accept() 方法接收数据。

在这里插入图片描述

光这个“纯裸”的 TCP 连接就可以收发数据了,那是不是就够了?

不行。

八股文常背 TCP 有三个特点:面向连接、可靠、基于字节流。

在这里插入图片描述

在这里插入图片描述

这里关注“基于字节流”,字节流可以理解为双向通道里流淌的二进制数据,就是一堆“01”串。“纯裸”的 TCP 连接收发的这些“01”串是没有任何边界的,你无法知道一条完整消息的起始位置。

在这里插入图片描述

由于无边界,所以当选择使用 TCP 发送“夏洛特烦恼”时,接收端无法区分你想表达的是“夏洛” 和“特烦恼”还是“夏洛特”和“烦恼”。这也就是所谓的“粘包”问题。

在这里插入图片描述

既然“纯裸” TCP 是不能直接拿来用的,那就需要在这个基础上加入一些自定义规则,来区分 消息边界。

于是,我们会把每条要发送的消息都包装一下。比如加入消息头,消息头中写清楚一个完整的包长度是多少,根据这个长度可以继续接收数据,截取出来后,就是我们真正想传输的消息体

这里的消息头中还可以放各种规则,比如消息体是否压缩、消息体格式之类的。只要上下游都约定好,互相能够解析就行了,这就是所谓的协议了。

在这里插入图片描述

每个使用 TCP 的项目,都可能会定义一套类似于这样的协议解析标准,它们可能有区别,但原理都类似。

于是基于 TCP 就衍生出非常多的协议,如 HTTP 和 RPC 。

TCP 是传输层的协议,而基于 TCP 造出来的各类协议,都是定义了不同规则的应用层协议。

在这里插入图片描述

HTTP 协议又叫做“超文本传输协议”,我们平时上网在浏览器中敲个网址就能访问网页,用到的就是 HTTP 协议。

在这里插入图片描述

RPC(Remote Procedure Call) 又叫做“远程过程调用”,它本身不是一种具体的协议,而是一种调用方式。

RPC 让我们可以像调用本地方法一样,调用远端服务器的方法(remoteFunc),屏蔽掉一些网络细节。

在这里插入图片描述

基于这个思路,大佬们造成了非常多款式的 RPC 协议,如 gRPC、tRPC 等。

虽然大部分 RPC 协议底层使用的是 TCP 协议,但也可以改用 UDP 或者 HTTP。

为什么有 HTTP 还要有 RPC?

其实,TCP 是 70 年代的产物,HTTP 是 90 年代的产物,由于直接使用 TCP 会出现粘包问题,所以可想而知,70~90 年代之间会有很多协议产生,其中包含 80 年代产出的 RPC 协议

在这里插入图片描述

因此我们该问的不是“为什么有 HTTP 了,还要有 RPC?”,而是“为什么有 RPC 了,还要有 HTTP?”。

为什么有 RPC 还要有 HTTP 协议?

现在电脑上安装的各种软件,如xx管家、xx视频,它们作为客户端,需要与服务端建立连接,收发消息,都会用到应用层协议。在这种 C/S(client server)架构下,他们就能使用自家创造的 RPC 协议,因为只需要连接自己公司的服务器。

但是xx浏览器却不同,不管是 Chrome 还是 IE,它们不仅需要连接自家服务器,还需要访问其他公司的网站服务器,因此需要一个统一的标准。于是 HTTP 就是当时用来统一 browser server 的协议。

当时 HTTP 主要用于 B/S 架构,而 RPC 主要用于 C/S 架构,但是现在没有分那么清楚了,B/S 和 C/S 在慢慢融合。

很多软件需要支持多端,如网页端、PC端、移动端,如果通信协议都用 HTTP,那么服务器只用一套就够了,而 RPC 开始退居幕后,一般用于公司内部集群里各个微服务之间的通讯。

那么,既然有了 HTTP,为啥还要使用 RPC 呢?

为什么有了 HTTP, 还要用 RPC?

这个问题需要从二者的区别说起:

首先是服务发现

服务建立连接的前提是需要知道对方的 IP 地址和端口号,寻找对方 IP 端口的过程叫做 服务发现。

在 HTTP 中,知道了服务域名,就可以通过 DNS 服务器解析 IP 地址,端口默认是 80 。

而 RPC 需要专门的中间服务来保存服务名的 IP 信息,比如 etcd (甚至是 Redis)。

可以看出服务发现方面,二者虽有区别,但不分高低。

在这里插入图片描述

其次是底层连接形式

以主流的 HTTP1.1 协议为例,默认在底层建立 TCP 连接之后会一直保持这个连接,Keep alive,之后的请求都会复用这条连接。

在这里插入图片描述

而 RPC 协议也和 HTTP 协议类似,也是通过建立 TCP 长连接进行数据交互。但不同的地方在于,RPC 一般还会再建立一个连接池,在请求量大的时候,建立多条连接放在连接池中,需要发数据的时候就从池中取出一条连接,用完再放回去。

在这里插入图片描述

由于连接池有利于提升网络性能,所以有些编程语言的网络库中也会给 HTTP 加个连接池,比如 Go。

可以看出这块二者也没有太大区别。

最后是传输的内容

HTTP 和 RPC 传输内容都是基于 TCP 传输的消息,包含消息头 header 和 消息体 body,header 适合标记一些特殊信息,其中最重要的信息就是消息体的长度;body 则是放真正需要传输的内容,而这些内容只能是二进制数据。

在这里插入图片描述

所以 TCP 传输字符串和数字的问题不大,因为字符串可以编码,再变成二进制,而数字本身可以直接转为二进制。

但是结构体呢?

需要想办法转为二进制,现在的方案主要有 json、protoBuf 等。将这个结构体转为二进制的过程叫做序列化。

在这里插入图片描述

在这里插入图片描述

对于主流的 HTTP1.1,虽然叫做超文本传输协议,支持音频视频,但是在设计之初是用于做网络文本展示的,所以传输内容主要以字符串为主,header 和 body 都是如此,在 body 这块使用 json 来序列化结构体。

而 RPC 因为定制化程度更高,可以采用体积更小的 protobuf 文件或其他序列化协议,来保存结构体,同时不需要像 HTTP 那样考虑各种浏览器行为,如 302 定向跳转,因此性能会更好。这也是公司内部抛弃 HTTP 而选择 RPC 的主要原因。

在这里插入图片描述

当然,上面说的 HTTP 其实特指主流的 HTTP1.1。

HTTP2 在前者的基础上做了很多改进,性能可能比 RPC 还好,甚至连 gRPC 底层都是用的 HTTP2,那么问题又来了,“为什么既然有了 HTTP2,还要使用 RPC?”

因为 HTTP2 是 2015 年才出来的,那时候很多公司内部的 RPC 服务都运行很多年了,基于历史原因,也就没必要换了。

参考:
RPC是什么?HTTP是什么?RPC和HTTP有什么区别?

备注:
在 HTTP/1.1 和 RPC(远程过程调用)中,消息头(header)和消息体(body)都是以二进制数据的形式在 TCP 连接上进行传输的。然而,它们的具体格式和编码方式有所不同。

  1. HTTP/1.1:

    • 消息头:HTTP 消息头是以文本格式编码的,通常是 ASCII 字符。每个头部字段由字段名和字段值组成,字段之间用冒号分隔。头部的结束是通过一个空行(即两个连续的换行符)来标识的。
    • 消息体:HTTP 消息体可以是任意类型的二进制数据(如图像、视频、JSON、XML 等),其内容的类型通常通过 Content-Type 头部字段来指明。消息体的长度可以通过 Content-Length 头部字段来指示,或者使用 Transfer-Encoding: chunked 进行分块传输。
  2. RPC:

    • RPC 的实现可以有多种形式(如 gRPC、XML-RPC、JSON-RPC 等),每种实现的消息格式可能不同。一般来说,RPC 的消息头和消息体也可以是二进制数据。
    • 在一些 RPC 实现中,消息头可能包含一些元数据(如方法名、请求 ID、版本信息等),而消息体则包含实际的参数或返回值。具体的编码方式(如 Protocol Buffers、Thrift 等)会影响消息的二进制表示。

总结来说,虽然 HTTP/1.1 的消息头是以文本格式传输的,但在底层 TCP 连接上,它们都是以二进制形式发送的。
而 RPC 的消息头和消息体通常都是以二进制格式传输的,具体取决于所使用的协议和编码方式。

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

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

相关文章

Y1打卡学习笔记

🍨 本文为🔗365天深度学习训练营 中的学习记录博客>- **🍖原作者:K同学啊** yolov5学习 下载源码运行命令查看结果视频检测个人总结 下载源码 地址:https://github.com/ultralytics/yolov5打开cmd后输入&#xff1…

【前端】Hexo 建站指南

文章目录 前言生成站点本地测试部署云端参考 前言 更好的阅读体验:https://blog.dwj601.cn/FrontEnd/Hexo/build-your-own-website-with-hexo/ 笔记记多了,想要分享给同学们一起交流进步,该怎么办?想要搭建一个属于自己的知识库…

【后端开发】字节跳动青训营之Go语言进阶与依赖管理

Go语言进阶与依赖管理 一、Go语言进阶1.1 并发与并行1.2 协程与线程1.3 通道1.3.1 生产消费模型 1.4 并发安全 二、依赖管理 一、Go语言进阶 Go语言一次可以创建上万个协程。 1.1 并发与并行 并发:多程序程序在单核CPU上运行。并行:多程序程序在多核CP…

大模型-本地化部署调用--基于ollama+openWebUI+springBoot

大模型-本地化部署调用–基于ollamaopenWebUIspringBoot 前言 前段时间,啊,可能不是前段时间,过去的2024年吧,大模型这块的内容也是非常火的,各家巨头也开始卷大模型的研发。那么本人呢也在过去的一年中也是用到了一…

RKNN_C++版本-YOLOV5

1.背景 为了实现低延时,所以开始看看C版本的rknn的使用,确实有不足的地方,请指正(代码借鉴了rk官方的仓库文件)。 2.基本的操作流程 1.读取模型初始化 // 设置基本信息 // 在postprocess.h文件中定义,详见…

H3C-防火墙IPSec配置案例(主模式)

目录 1.IPSec简述:2.IPSec应用场景:3.网络拓扑及说明:4.案例背景:5.网络配置:5.1 基础网络配置:5.1.1 总部防火墙基础配置:5.1.2 分部防火墙基础配置:5.1.3 互联网路由器基础配置:5.1.4 总部服务器基础配置:5.1.5 总部PC基础配置: 5.2 IPSec配置:5.2.1 总部防火墙IPSec配置:5.2…

windows下本地部署安装hadoop+scala+spark-【不需要虚拟机】

注意版本依赖【本实验版本如下】 Hadoop 3.1.1 spark 2.3.2 scala 2.11 1.依赖环境 1.1 java 安装java并配置环境变量【如果未安装搜索其他教程】 环境验证如下: C:\Users\wangning>java -version java version "1.8.0_261" Java(TM) SE Runti…

vim如何显示行号

:set nu 显示行号 :set nonu 不显示行号 (vim如何使设置显示行号永久生效:vim如何使相关设置永久生效-CSDN博客)

国产编辑器EverEdit - 命令窗口应用详解

1 命令窗口应用详解 1.1 应用场景 有时需要在EverEdit中执行一些命令行工具,甚至想把当前文档做为参数,传递给命令进行一些文本分析,比如:一些常用的文本处理工具,gawk.exe等。 1.2 使用方法 命令窗口的使用在官方手…

Linux C\C++编程-文件位置指针与读写文件数据块

【图书推荐】《Linux C与C一线开发实践(第2版)》_linux c与c一线开发实践pdf-CSDN博客 《Linux C与C一线开发实践(第2版)(Linux技术丛书)》(朱文伟,李建英)【摘要 书评 试读】- 京东图书 Linu…

vue2使用flv.js在浏览器打开flv格式视频

组件地址&#xff1a;GitHub - bilibili/flv.js: HTML5 FLV Player flv.js 仅支持 H.264 和 AAC/MP3 编码的 FLV 文件。如果视频文件使用了其他编码格式就打不开。 flv.vue <template><div><el-dialog :visible.sync"innerVisibleFlv" :close-on-pre…

Linux下Ubuntun系统报错find_package(BLAS REQUIRED)找不到

Linux下Ubuntun系统报错find_package(BLAS REQUIRED)找不到 这次在windows的WSL2中遇到了一个非常奇怪的错误&#xff0c;就是 CMake Error at /usr/share/cmake-3.22/Modules/FindPackageHandleStandardArgs.cmake:230 (message):Could NOT find BLAS (missing: BLAS_LIBRAR…

仿 RabbitMQ 的消息队列3(实战项目)

七. 消息存储设计 上一篇博客已经将消息统计文件的读写代码实现了&#xff0c;下一步我们将实现创建队列文件和目录。 实现创建队列文件和目录 初始化 0\t0 这样的初始值. //创建队列对应的文件和目录&#xff1a;public void createQueueFile(String queueName) throws IO…

多线程杂谈:惊群现象、CAS、安全的单例

引言 本文是一篇杂谈&#xff0c;帮助大家了解多线程可能会出现的面试题。 目录 引言 惊群现象 结合条件变量 CAS原子操作&#xff08;cmp & swap&#xff09; 线程控制&#xff1a;两个线程交替打印奇偶数 智能指针线程安全 单例模式线程安全 最简单的单例&…

腾讯 Hunyuan3D-2: 高分辨率3D 资产生成

腾讯 Hunyuan3D-2&#xff1a;高分辨率 3D 资产生成的突破 前言 在当今数字化时代&#xff0c;3D 资产生成技术正变得越来越重要。无论是游戏开发、影视制作还是虚拟现实领域&#xff0c;高质量的 3D 模型和纹理都是创造沉浸式体验的关键。然而&#xff0c;传统的 3D 资产制作…

R语言学习笔记之开发环境配置

一、概要 整个安装过程及遇到的问题记录 操作步骤备注&#xff08;包含遇到的问题&#xff09;1下载安装R语言2下载安装RStudio3离线安装pacman提示需要安装Rtools4安装Rtoolspacman、tidyfst均离线安装完成5加载tidyfst报错 提示需要安装依赖&#xff0c;试错逐步下载并安装…

DRG/DIP 2.0时代下基于PostgreSQL的成本管理实践与探索(上)

一、引言 1.1 研究背景与意义 在医疗领域的改革进程中&#xff0c; DRG/DIP 2.0 时代&#xff0c;医院成本管理的重要性愈发凸显。新的医保支付方式下&#xff0c;医院的收入不再单纯取决于医疗服务项目的数量&#xff0c;而是与病种的分组、费用标准以及成本控制紧密相关。这…

【数据结构】_顺序表

目录 1. 概念与结构 1.1 静态顺序表 1.2 动态顺序表 2. 动态顺序表实现 2.1 SeqList.h 2.2 SeqList.c 2.3 Test_SeqList.c 3. 顺序表性能分析 线性表是n个具有相同特性的数据元素的有限序列。 常见的线性表有&#xff1a;顺序表、链表、栈、队列、字符串等&#xff1b…

缓存之美:万文详解 Caffeine 实现原理(下)

上篇文章&#xff1a;缓存之美&#xff1a;万文详解 Caffeine 实现原理&#xff08;上&#xff09; getIfPresent 现在我们对 put 方法有了基本了解&#xff0c;现在我们继续深入 getIfPresent 方法&#xff1a; public class TestReadSourceCode {Testpublic void doRead() …

VSCode下EIDE插件开发STM32

VSCode下STM32开发环境搭建 本STM32教程使用vscode的EIDE插件的开发环境&#xff0c;完全免费&#xff0c;有管理代码文件的界面&#xff0c;不需要其它IDE。 视频教程见本人的 VSCodeEIDE开发STM32 安装EIDE插件 Embedded IDE 嵌入式IDE 这个插件可以帮我们管理代码文件&am…