使用Golang Net/RPC包实现简单的同步通信_标准库版远程调用

1次阅读

net/rpc 默认只支持导出方法,因反射无法访问小写开头的未导出方法;方法签名须为 func(t, args, *reply) Error;默认 gob 编码不跨语言;超时需用 context 包裹调用。

使用Golang Net/RPC包实现简单的同步通信_标准库版远程调用

为什么 net/rpc 默认只支持导出方法

因为 Go 的 RPC 服务端靠反射遍历结构体方法,未导出(小写开头)的方法在包外不可见,server.register 根本看不到它们,调用时直接报 method not found

实操建议:

立即学习go语言免费学习笔记(深入)”;

  • 服务端注册的结构体字段和方法名必须首字母大写(如 UserGetInfo
  • 方法签名必须严格为 func(*T, *Args, *Reply) error,三个参数都得是指针,且第三个是输出结果
  • 如果传值类型(如 int)当 *Args,会 panic:「rpc: method has wrong number of ins」

gob 编解码限制导致跨语言调用失败

net/rpc 默认用 gob 序列化,它依赖 Go 类型元信息,其他语言根本解析不了 —— 这不是 bug,是设计使然。想和 Python/js 通信?别用标准库 net/rpc,换 jsonrpc2 或 gRPC。

实操建议:

立即学习go语言免费学习笔记(深入)”;

  • 纯 Go 内部通信可用默认 gob,但客户端和服务端 Struct 字段标签(如 json:"id")不影响 gob,它只看字段名和顺序
  • 若手动换 json 编解码,需调用 rpc.NewServer() 后用 server.RegisterCodec,且客户端也得配对使用 jsonrpc.NewClient
  • 常见错误现象:unexpected EOFinvalid character,大概率是客户端没切 codec,还在发 gob 流

阻塞式调用下超时控制必须靠 context 手动包一层

net/rpcClient.CallGo 方法本身不接受 context.Context,超时只能靠外部控制。直接设 conn.SetDeadline 不行 —— RPC 是多请求复用连接的,改 deadline 会影响后续调用。

实操建议:

立即学习go语言免费学习笔记(深入)”;

  • context.WithTimeout 包住整个调用逻辑,再起 goroutine + select 等待结果或超时
  • 不要试图给底层 net.Conn 设读写 deadline,RPC 消息头+体是分步读的,中间 deadline 到期会导致粘包错乱
  • 示例关键片段:
    ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second) defer cancel() done := make(chan *rpc.Call, 1) go func() {     done <- client.Go("User.GetInfo", &args, &reply, nil) }() select { case call := <-done:     if call.Error != nil { /* 处理错误 */ } case <-ctx.Done():     /* 超时,但注意:call 可能还在跑,无法取消 */ }

服务端启动后没监听到请求?检查 http.Serve 是否误用了

net/rpc 本身不带 HTTP 服务逻辑,但文档里那个 http.Serve 示例容易误导人 —— 它只是把 RPC handler 挂到 HTTP server 上,走的是 HTTP POST + gob body。如果你没配好 HTTP router 或忘了 http.Handle,请求就静默丢弃,连日志都没有。

实操建议:

立即学习go语言免费学习笔记(深入)”;

  • 最简 TCP 直连方式:用 rpc.ServeConnnet.Listen,绕过 HTTP 层,调试更直接
  • 如果坚持走 HTTP,必须显式注册:http.Handle("/RPC2", rpc.DefaultServer),路径名不能错,且客户端 URL 得匹配(如 http://localhost:8080/RPC2
  • 常见错误现象:curl 测试返回 404 或空响应,tcpdump 看到 SYN-ACK 但无数据 —— 八成是 handler 没挂对路径

真正麻烦的从来不是写通一次调用,而是当多个客户端并发注册、方法返回 error 但 reply 已部分写入、或者服务端 panic 后连接卡死 —— 这些边界在标准库里全得自己兜底。

text=ZqhQzanResources