0

0

如何在Golang中实现gRPC通信_Golang gRPC服务开发与实践

P粉602998670

P粉602998670

发布时间:2026-02-06 13:04:02

|

133人浏览过

|

来源于php中文网

原创

gRPC服务默认不支持HTTP/1.1,需通过h2c启用HTTP/2明文通信;curl需加--http2且服务端用h2c.NewHandler包装;客户端DeadlineExceeded多因拦截器阻塞或超时设置过短;proto变更需手动触发go generate。

如何在golang中实现grpc通信_golang grpc服务开发与实践

Go 的 gRPC 服务默认不支持 HTTP/1.1 请求,直接用浏览器访问 /healthcurl GET 会返回 404 Not FoundHTTP/2 stream error —— 这不是代码写错了,是协议层根本对不上。

为什么 grpc.Server 启动后 curl 报错 “connection refused” 或 “protocol error”

gRPC 基于 HTTP/2,而 curl 默认走 HTTP/1.1;即使加 --http2,也需配合 TLS(或显式启用 h2c)才能连上非加密 gRPC 端口。常见表现:

  • 未启用 h2c 时,curl -v http://localhost:8080 直接被拒绝或返回空响应
  • grpcurl 测试却提示 Failed to dial target host,大概率是没开 Reflection 或端口被防火墙拦截
  • Go 服务监听 :0(随机端口)但没打印实际绑定地址,导致客户端连错

实操建议:
- 启动服务时强制指定端口并打印日志:log.Printf("gRPC server listening at %v", lis.Addr())
- 非生产环境快速验证,启用 h2c:grpcServer := grpc.NewServer(grpc.ProtocolVersion(1)) 不行 —— 正确做法是在 http.Server 中用 h2c.NewHandler() 包裹 gRPC handler(见下条)

如何让 gRPC Server 同时支持 h2c(HTTP/2 without TLS)

Go 标准库的 net/http 默认不支持 h2c,必须手动桥接。核心是把 grpc.Server 注册为 http.Handler,再用 h2c 中间件透传 HTTP/2 流量。

关键步骤:

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

Pebblely
Pebblely

AI产品图精美背景添加

下载
  • 导入 "golang.org/x/net/http2/h2c"
  • 创建 http.Server,将 grpcServer.ServeHTTP 作为 handler:handler := h2c.NewHandler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { grpcServer.ServeHTTP(w, r) }), &h2c.Server{})
  • 监听时不要用 grpcServer.Serve(lis),改用 httpServer.Serve(lis)
  • 注意:此时不能再调用 grpcServer.GracefulStop(),需同步控制 http.Server 的生命周期

这样之后,就能用 curl --http2 -X POST http://localhost:8080/my.Service/Method -d '{"id":1}'(需 JSON mapping)或 grpcurl -plaintext localhost:8080 list 直接调试。

客户端连接时 context.DeadlineExceeded 的真实原因

这个错误常被误认为网络不通,其实多数情况是服务端未及时响应 —— 尤其在启用了拦截器(interceptor)、中间件或鉴权逻辑阻塞时。典型场景:

  • 服务端 UnaryInterceptor 里做了同步 HTTP 调用,且对方响应慢
  • 客户端 grpc.WithTimeout(1 * time.Second) 设得太短,而服务端处理耗时 1.2s
  • DNS 解析失败或 resolver 配置错误,导致连接卡在 Connecting 状态超时

排查建议:
- 客户端加日志:用 grpc.WithStatsHandler(&customStatsHandler{}) 观察各阶段耗时
- 服务端拦截器开头打时间戳,确认是否卡在业务逻辑前
- 检查 resolver 是否用了自定义方案(如 dns:/// 或 etcd),避免解析延迟

Proto 文件变更后,为什么 go generate 不触发重新生成 pb.go

Go 的 //go:generate 是静态指令,不会自动感知 .proto 文件修改。常见陷阱:

  • 执行 go generate ./... 时,当前目录下没有 go:generate 注释,导致跳过
  • protoc 命令路径未加入 $PATH,或 protoc-gen-go 版本与 google.golang.org/protobuf 不兼容(如 v1.31+ 需要 protoc-gen-go v1.32+)
  • import 路径写成相对路径(如 import "proto/common.proto"),但 protoc --proto_path=. 未覆盖该目录

可靠做法:
- 在 api/ 目录下放一个 generate.go,含完整 //go:generate protoc --go_out=paths=source_relative:. --go-grpc_out=paths=source_relative:. *.proto
- 提交前跑一次 make proto(Makefile 封装命令),比依赖 IDE 自动生成更可控

最易忽略的一点:gRPC 的流式接口(stream)一旦客户端提前关闭 context,服务端 Send() 可能 panic,必须用 if err != nil && status.Code(err) == codes.Canceled 显式判断,而不是只看 err != nil

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

231

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

344

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

398

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

282

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

196

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

621

2025.06.17

1688阿里巴巴货源平台入口与批发采购指南
1688阿里巴巴货源平台入口与批发采购指南

本专题整理了1688阿里巴巴批发进货平台的最新入口地址与在线采购指南,帮助用户快速找到官方网站入口,了解如何进行批发采购、货源选择以及厂家直销等功能,提升采购效率与平台使用体验。

21

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.9万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号