通八洲科技

如何使用Golang实现RPC序列化与反序列化_Golang RPC数据编码与解码方法

日期:2026-01-01 00:00 / 作者:P粉602998670
net/rpc 默认用 gob,不支持跨语言;换 JSON 需用 jsonrpc 包并调用 jsonrpc.ServeConn;换 protobuf 应直接用 gRPC;自定义 ServerCodec 必须处理分帧、Header 解析和错误传播。

Go 标准库的 net/rpc 默认使用 gob 编码,不支持跨语言、不兼容 JSON 或 Protocol Buffers——如果你需要自定义序列化(比如用 jsonprotobuf),必须绕过默认机制,自己接管编解码流程。

为什么不能直接替换 rpc.Server 的编解码器?

标准 rpc.Serverrpc.Client 将编码/解码深度耦合在 rpc.Server.ServeCodecrpc.NewClientWithCodec 中,底层依赖 rpc.ServerCodec 接口。它要求你同时实现 ReadRequestHeader/ReadRequestBodyWriteResponse 等方法,不能只换一个序列化格式。

常见错误是试图“简单替换 gobjson”,结果发现请求头读不出来、响应乱序、或连接直接断开——因为 json 没有 gob 那样的类型元信息和流式分帧能力。

如何用 json 实现兼容标准 rpc.Client 的服务端?

使用 net/rpc/jsonrpc 是最轻量、最兼容的方式:它复用了标准 rpc 的注册与调用逻辑,仅替换底层编解码器。客户端无需改代码,只要把 rpc.NewClient 换成 jsonrpc.NewClient 即可。

服务端启动示例:

package main

import ( "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 } }

注意点:

如何用 protobuf + gRPC 替代原生 net/rpc

如果你真正想要的是高性能、跨语言、带 IDL 的 RPC,不要硬改 net/rpc,直接用 gRPC-Go。它本质是基于 HTTP/2 的二进制 RPC 框架,protobuf 是其默认序列化协议,天然支持 streaming、拦截器、超时、认证等。

关键区别:

所以,除非你有强约束(如必须复用现有 net/rpc 客户端、不能引入新协议栈),否则别尝试“给 net/rpc 换 protobuf 底层”。真要这么做,得完整实现 rpc.ServerCodec,包括:ReadRequestHeader 解析 method/service/seq,ReadRequestBodyproto.Unmarshal,还要处理 length-prefix framing——这已接近重写整个 RPC 栈。

自定义编解码器时最容易忽略的细节

当你实现自己的 rpc.ServerCodec,以下三点几乎必踩坑:

真正稳定的自定义方案,往往不是从零写 ServerCodec,而是基于 gRPCCodec 接口或 go-microcodec 插件机制——它们已封装好分帧、上下文、错误传播等细节。