如何使用Golang实现RPC序列化与反序列化_Golang RPC数据编码与解码方法
技术百科
P粉602998670
发布时间:2026-01-01
浏览: 次 net/rpc 默认用 gob,不支持跨语言;换 JSON 需用 jsonrpc 包并调用 jsonrpc.ServeConn;换 protobuf 应直接用 gRPC;自定义 ServerCodec 必须处理分帧、Header 解析和错误传播。
Go 标准库的 net/rpc 默认使用 gob 编码,不支持跨语言、不兼容 JS
ON 或 Protocol Buffers——如果你需要自定义序列化(比如用 json 或 protobuf),必须绕过默认机制,自己接管编解码流程。
为什么不能直接替换 rpc.Server 的编解码器?
标准 rpc.Server 和 rpc.Client 将编码/解码深度耦合在 rpc.Server.ServeCodec 和 rpc.NewClientWithCodec 中,底层依赖 rpc.ServerCodec 接口。它要求你同时实现 ReadRequestHeader/ReadRequestBody 和 WriteResponse 等方法,不能只换一个序列化格式。
常见错误是试图“简单替换 gob 为 json”,结果发现请求头读不出来、响应乱序、或连接直接断开——因为 json 没有 gob 那样的类型元信息和流式分帧能力。
-
gob自带类型描述、支持指针/接口/循环引用,且天然适配 Go 运行时;json是纯文本、无类型、需显式结构体标签 - RPC 协议不是“发一串 JSON 就完事”,它需要严格区分 header(含方法名、序列号)和 body(参数),而
json不提供内置分界机制 - 标准
rpc/jsonrpc包只是对gob编解码逻辑的重写,并非“用encoding/json替代gob”——它仍遵循 RPC 帧格式约定
如何用 json 实现兼容标准 rpc.Client 的服务端?
使用 net/rpc/jsonrpc 是最轻量、最兼容的方式:它复用了标准 rpc 的注册与调用逻辑,仅替换底层编解码器。客户端无需改代码,只要把 rpc.NewClient 换成 jsonrpc.NewClient 即可。
服务端启动示例:
package mainimport ( "net" "net/rpc" "net/rpc/jsonrpc" )
type Args struct{ A, B int } type Quotient struct{ Quo, Rem int }
type Arith int
func (t Arith) Multiply(args Args, reply int) error { reply = args.A * args.B return nil }
func main() { rpc.Register(new(Arith)) listener, := net.Listen("tcp", ":1234") for { conn, := listener.Accept() go jsonrpc.ServeConn(conn) // ← 关键:用 jsonrpc.ServeConn 替代 rpc.ServeConn } }
注意点:
- 必须用
jsonrpc.ServeConn启动连接,不能用rpc.ServeConn - 客户端必须用
jsonrpc.NewClient或jsonrpc.Dial,否则会因帧格式不匹配而报invalid character错误 - 结构体字段必须首字母大写(导出),且建议加
json:标签以控制 key 名
如何用 protobuf + gRPC 替代原生 net/rpc?
如果你真正想要的是高性能、跨语言、带 IDL 的 RPC,不要硬改 net/rpc,直接用 gRPC-Go。它本质是基于 HTTP/2 的二进制 RPC 框架,protobuf 是其默认序列化协议,天然支持 streaming、拦截器、超时、认证等。
关键区别:
-
net/rpc是 TCP + 自定义二进制帧(gob)或 JSON 帧;gRPC是 HTTP/2 +protobuf+gRPC-encoding(length-delimited) -
gRPC要求先写.proto文件生成 Go 代码;net/rpc直接注册结构体和方法 - 没有
gRPC服务端,protobuf只是序列化工具——你仍需自己设计传输层(比如用net.Conn+ 自定义分帧),这极易出错
所以,除非你有强约束(如必须复用现有 net/rpc 客户端、不能引入新协议栈),否则别尝试“给 net/rpc 换 protobuf 底层”。真要这么做,得完整实现 rpc.ServerCodec,包括:ReadRequestHeader 解析 method/service/seq,ReadRequestBody 用 proto.Unmarshal,还要处理 length-prefix framing——这已接近重写整个 RPC 栈。
自定义编解码器时最容易忽略的细节
当你实现自己的 rpc.ServerCodec,以下三点几乎必踩坑:
- TCP 是字节流,没有消息边界——必须自己加长度前缀(如 4 字节 uint32 表示后续 payload 长度),否则
ReadRequestBody会阻塞或读错数据 -
ReadRequestHeader必须严格按 RPC 协议解析出ServiceMethod字符串(格式为"Service.Method")和Seq(请求序号),漏掉任一字段会导致客户端永远收不到响应 -
WriteResponse的err参数不能忽略:如果业务逻辑返回 error,必须写入 response body 并设置ErrorResponse标志,否则客户端Call会卡死或 panic
真正稳定的自定义方案,往往不是从零写 ServerCodec,而是基于 gRPC 的 Codec 接口或 go-micro 的 codec 插件机制——它们已封装好分帧、上下文、错误传播等细节。
# ai
# 如果你
# 如何用
# 重写
# 自定义
# 客户端
# 不支持
# 工具
# http
# js
# json
# go
# golang
# 循环
# Error
# 编码
# 字节
# 区别
# 标准库
# 指针
# stream
# 字符串
# 接口
# 序列化
# 为什么
# 无类型
# 栈
# 封装
# 结构体
# rpc
# Length
# 如用
# 服务端
# 编解码器
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11怎么关闭专注助手 Win11关闭免打扰模
- C++中的std::shared_from_thi
- Win10电脑怎么设置IP地址_Windows10
- Win11怎么更改系统语言为中文_Windows1
- 如何在Golang中实现WebSocket广播_使
- Win11怎么开启游戏工具栏_Windows11
- Python对象生命周期管理_创建销毁解析【教程】
- Win11怎么更改默认打开方式_Win11关联文件
- windows系统找不到无线网络怎么办_windo
- Python 模块的 __name__ 属性如何由
- Win11摄像头无法使用怎么办_Win11相机隐私
- Win11怎么清理C盘OneDrive缓存_Win
- c++中如何使用虚函数实现多态_c++多态性实现原
- Win11此电脑不在桌面上_Windows 11桌
- C++友元类使用场景_C++类间协作设计方式讲解
- Win11如何设置鼠标灵敏度_Win11鼠标灵敏度
- c++输入输出流 c++ cin与cout格式化输
- Python实现图数据库操作_Neo4j核心CRU
- Linux如何使用grep搜索文件内容_Linux
- 如何在 Go 结构体中正确初始化 map 字段
- 如何在 Windows 11 中使用 AlomWa
- mac怎么安装pip_MAC Python pip
- php怎么捕获异常_trycatch结构处理运行时
- 如何使用Golang安装API文档生成工具_快速生
- 如何使用Golang开发简单的聊天室消息存储_Go
- 怎么将XML数据可视化 D3.js加载XML
- 如何从 Go 的 map[string]inter
- Win11怎么关闭边缘滑动手势_Windows11
- Win11怎么关闭OneDrive同步_Win11
- Mac怎么给文件夹加密_Mac创建加密磁盘映像教程
- TestNG的testng.xml配置文件怎么写
- php内存溢出怎么排查_php内存限制调试与优化方
- 如何使用Golang反射创建map对象_动态生成键
- php删除数据怎么软删除_添加is_del字段标记
- Python音视频处理高级项目教程_FFmpegP
- 如何使用Golang实现RPC序列化与反序列化_G
- Win10如何卸载WindowsDefender_
- 一文详解网站被黑客入侵挂马解决办法
- Win10怎样卸载iTunes_Win10卸载iT
- 如何在Golang中实现文件下载_Golang文件
- Python集合操作技巧_高效去重解析【教程】
- Windows10如何更改日期格式_Win10区域
- Win10怎么卸载金山毒霸_Win10彻底卸载金山
- Python深度学习实战教程_神经网络模型构建与训
- 如何在Golang中使用replace替换模块_指
- Windows如何使用BitLocker To G
- 如何使用Golang理解结构体指针方法接收者_Go
- Go语言中slice追加操作的底层共享机制详解
- Win11如何设置计划任务 Win11定时执行程序
- Win11怎么更改计算机名_Windows11系统

QQ客服