当前位置: 首页 > news >正文

【知识】RPC和gRPC

转载请注明出处:小锋学长生活大爆炸[xfxuezhagn.cn]

如果本文帮助到了你,欢迎[点赞、收藏、关注]哦~

RPC

1. RPC 的定义

RPC(Remote Procedure Call,远程过程调用) 是一种进程间通信协议。它允许程序调用另一台机器(或同一台机器的其他进程)上的函数/过程,好像在调用本地函数一样,隐藏了网络通信的细节。

核心思想:

  • 调用者像调用本地函数一样调用远程函数。

  • 底层通过序列化(编码)请求、网络传输、远程解码执行,再将结果返回。

2. RPC 的基本流程

  1. 客户端调用本地代理(Stub)方法。

  2. Stub 将方法名、参数等序列化(通常称为 封送/marshalling)。

  3. 序列化后的数据通过网络发送到服务端。

  4. 服务端收到后,反序列化(解封装),调用真正的服务函数。

  5. 服务函数执行完,将结果序列化,通过网络返回。

  6. 客户端 Stub 收到后反序列化,返回结果。

👉 即:Client -> stub -> network -> server -> real function -> return

3. RPC 的优点

  • 把远程调用封装得像本地函数调用一样,降低了分布式编程的复杂度。

  • 可以跨语言、跨平台。

  • 能在微服务架构中实现服务间解耦。

4. 实现示例

这是使用 socket + pickle 来模拟最原始的 RPC,帮助理解核心原理。

服务端(Server)

# server.py
import socket
import pickledef add(x, y):return x + ydef handle_request(data):func_name, args = pickle.loads(data)if func_name == 'add':return pickle.dumps(add(*args))return pickle.dumps("Unknown function")server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(('localhost', 8000))
server.listen(5)
print("RPC Server running on port 8000...")while True:client, addr = server.accept()data = client.recv(1024)result = handle_request(data)client.send(result)client.close()

客户端(Client)

# client.py
import socket
import pickledef remote_call(func_name, args):client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)client.connect(('localhost', 8000))client.send(pickle.dumps((func_name, args)))data = client.recv(1024)client.close()return pickle.loads(data)print("Calling remote add(5, 7)...")
result = remote_call('add', (5, 7))
print("Result:", result)

启动方式

python server.py
# 另开终端
python client.py

输出结果

Calling remote add(5, 7)...
Result: 12

gRPC

1. gRPC 的定义

gRPC 是 Google 开源的高性能、通用的 RPC 框架,用来在分布式系统中进行服务间通信。它基于:

  • HTTP/2(高效双向流传输,支持多路复用、头部压缩、流控)

  • Protocol Buffers(protobuf)(高效序列化)

2. gRPC 的特点

  • 多语言支持:官方支持 Go、Java、C++、Python、C#、Node.js、PHP、Ruby 等。
  • IDL(接口定义语言)统一规范:.proto 文件定义服务接口和消息结构。自动生成客户端 & 服务端代码。
  • 高效序列化:使用 protobuf,比 JSON / XML 更紧凑、更快。
  • 基于 HTTP/2:
    • 支持多路复用(一个 TCP 连接上多条流),减少连接数量

    • 头部压缩减少带宽

    • 支持流式数据(Client Streaming、Server Streaming、Bi-directional Streaming)

  • 内置健康检查、超时、重试、负载均衡、认证

3. gRPC 的工作原理

  1. 使用 .proto 文件定义服务和消息。

  2. 通过 protoc 生成客户端 & 服务端代码。

  3. 客户端通过生成的 Stub 调用服务端,就像调用本地方法。

4. 使用示例

下面是一个标准的 gRPC 示例,用 .proto 文件定义接口,然后生成 Python 代码并运行。

定义 proto 文件:新建 helloworld.proto

syntax = "proto3";package helloworld;service Greeter {rpc SayHello (HelloRequest) returns (HelloReply);
}message HelloRequest {string name = 1;
}message HelloReply {string message = 1;
}

生成 Python 代码

python -m grpc_tools.protoc -I. --python_out=. --grpc_python_out=. helloworld.proto

会生成:

  • helloworld_pb2.py (消息类)

  • helloworld_pb2_grpc.py (服务类)

实现服务端:新建 server.py

from concurrent import futures
import grpc
import helloworld_pb2
import helloworld_pb2_grpcclass Greeter(helloworld_pb2_grpc.GreeterServicer):def SayHello(self, request, context):return helloworld_pb2.HelloReply(message=f"Hello, {request.name}")def serve():server = grpc.server(futures.ThreadPoolExecutor(max_workers=10))helloworld_pb2_grpc.add_GreeterServicer_to_server(Greeter(), server)server.add_insecure_port('[::]:50051')server.start()print("gRPC server running on port 50051...")server.wait_for_termination()if __name__ == '__main__':serve()

实现客户端:新建 client.py

import grpc
import helloworld_pb2
import helloworld_pb2_grpcdef run():with grpc.insecure_channel('localhost:50051') as channel:stub = helloworld_pb2_grpc.GreeterStub(channel)response = stub.SayHello(helloworld_pb2.HelloRequest(name="Alice"))print("Greeter client received:", response.message)if __name__ == '__main__':run()

启动方式

python server.py
# 另开终端
python client.py

输出结果

Greeter client received: Hello, Alice

RPC 与 gRPC 的关系与区别

维度RPC (泛指)gRPC(具体实现)
概念一种通信思想 / 模式Google 实现的开源高性能 RPC 框架
传输协议不固定,可用 HTTP/TCP 等固定基于 HTTP/2
序列化XML/JSON/自定义二进制等Protobuf(默认)
语言支持取决于具体 RPC 框架官方支持多语言
IDL可以是 Thrift/IDL/自定义Proto 文件(.proto)
性能通常依赖实现gRPC 性能优异
流式支持不一定原生支持流式(stream)调用
生态工具分散,因框架而异gRPC 官方提供丰富工具链

http://www.lryc.cn/news/578067.html

相关文章:

  • Reactor操作符的共享与复用
  • Excel数据匹配合并工具
  • Linux 系统管理:自动化运维与容器化部署
  • 2025年数字信号、计算机通信与软件工程国际会议(DSCCSE 2025)
  • postman接口测试全部流程
  • Git 简介安装教程
  • [附源码+数据库+毕业论文]基于Spring+MyBatis+MySQL+Maven+jsp实现的校园服务平台管理系统,推荐!
  • Fiddler中文版抓包工具如何帮助前端开发者高效调试
  • 我的第一个开源项目:用Python搭建轻量级静态网页服务器—— 零基础也能实现的Web开发初体验
  • 鸿蒙应用开发:ArkTS中接口的声明和使用
  • SQL优化(插入、主键、order by、group by)
  • 关于 java:8. Java 内存模型与 JVM 基础
  • ClickHouse 部署
  • RK3568平台开发系列讲解:WIFI的调试手段
  • 重构老项目不再“踩雷”:飞算JavaAI的本地化智能合并实战
  • 企业自建云概念解读|私有云、专有云、混合云、分布式云、企业云
  • Windows桌面上的「了解此图片」怎么弄掉?
  • Tailwind CSS 配置正确,也没有报错,但是样式没有生效(解决~)
  • 如何用废弃电脑变成服务器搭建web网站(公网访问零成本)
  • 成像光谱遥感技术中的AI革命:ChatGPT在遥感领域中的应用
  • 【MySQL基础】MySQL索引全面解析:从原理到实践
  • vscode ssh远程连接ubuntu20失败的解决方法
  • 第9篇:Gin配置管理-Viper的实战使用
  • 批量生成文件名6
  • LeetCode Hot100 (二叉树)
  • 大模型-分布式推理简介
  • Deepoc 大模型在无人机行业应用效果的方法
  • c# 详细分析Task.sleep和Thread.sleep 的区别、使用场景和应用示例
  • MySQL中查询JSON数组字段包含特定字符串的方法
  • 通过python+openCV实现对图片中箭头方向的判断