0

0

c++如何使用gRPC实现高性能RPC服务? (Protobuf基础)

尼克

尼克

发布时间:2026-01-16 10:20:03

|

981人浏览过

|

来源于php中文网

原创

Protobuf文件必须首行声明syntax="proto3",gRPC C++仅支持proto3;字段默认optional,枚举首值须为0;需统一protoc版本生成代码,服务端用BuildAndStart()启动,客户端高并发应使用异步API与复用CompletionQueue。

c++如何使用grpc实现高性能rpc服务? (protobuf基础)

Protobuf 文件定义必须显式指定 syntax = "proto3"

很多初学者在写 .proto 文件时漏掉语法声明,导致 protoc 编译失败并报错 Expected "syntax = ..." 。gRPC C++ 默认只支持 proto3,且不兼容 proto2 的默认值、required 字段等行为。

  • syntax = "proto3"; 必须是文件第一行(可空行,但不可注释)
  • 所有字段默认为 optional,无需加 optional 关键字(加了会报错)
  • 枚举第一个值必须为 0,否则反序列化时可能被当为未设置
  • 使用 google/protobuf/wrappers.proto 中的 Int32Value 等才能表达“空值”语义
syntax = "proto3";

package example;

message Request {
  string user_id = 1;
  int32 timeout_ms = 2;
}

message Response {
  bool success = 1;
  string message = 2;
}

service Greeter {
  rpc SayHello(Request) returns (Response);
}

C++ 客户端和服务端代码必须用 protoc 生成相同的头文件和源文件

手动改写或复用旧版生成代码极易引发 ABI 不匹配:比如服务端用 protobuf 3.21 编译,客户端用 3.19 生成的 *.grpc.pb.h,会导致 Unknown field in serialization 或崩溃。

  • 始终用同一版本 protoc(推荐与 libprotobuf-dev 同版本)执行生成
  • 生成命令必须包含 --grpc_out--cpp_out,且顺序不能颠倒
  • 编译时需同时链接 libgrpc++libprotobuflibgrpc —— 少一个就链接失败或运行时 undefined symbol
  • 确保 #include 路径包含生成目录,且优先级高于系统 protobuf 头(避免混用)

服务端启动必须调用 ServerBuilder::BuildAndStart(),不能只调用 Build()

常见错误是只调用 builder.Build(),结果程序无报错但监听端口不生效、客户端连接超时。这是因为 Build() 只返回 std::unique_ptr,但没真正启动 I/O 循环。

因赛AIGC
因赛AIGC

因赛AIGC解决营销全链路应用场景

下载
  • BuildAndStart() 内部会启动 gRPC 的 completion queue 线程池,默认线程数为 CPU 核心数
  • 若需控制并发,应通过 SetMaxMessageSize()SetResourceQuota() 限制,而非减少线程数
  • 服务端 shutdown 必须先调用 server->Shutdown(),再 wait_for_idle(),否则可能丢弃正在处理的请求
grpc::ServerBuilder builder;
builder.AddListeningPort("0.0.0.0:50051", grpc::InsecureServerCredentials());
builder.RegisterService(&service);
std::unique_ptr server = builder.BuildAndStart(); // 注意是 BuildAndStart
std::cout << "Server listening on 0.0.0.0:50051\n";
server->Wait();

客户端同步调用易阻塞主线程,高并发场景务必用异步 API + CompletionQueue

stub->SayHello(&context, request, &response) 这种同步方式,在 QPS > 100 时容易因网络延迟堆积线程,最终耗尽内存或触发 OS 线程创建失败。

立即学习C++免费学习笔记(深入)”;

  • 异步模式下,每个 RPC 调用只占少量空间,靠 CompletionQueue::Next() 统一收包
  • 不要为每个请求 new 一个 CompletionQueue;推荐复用 1–4 个全局队列,按负载绑定线程
  • 注意 ClientContext 生命周期:必须在 AsyncNext() 返回前保持有效,否则回调中访问 context 成员会 crash
  • 超时必须设在 ClientContext::set_deadline(),而不是靠外部 timer —— gRPC 内部 deadline 检查更精确
C++ 的 gRPC 高性能关键不在“怎么写”,而在“怎么不写错”:protobuf 版本对齐、生成代码全链路一致、completion queue 使用粒度、deadline 和生命周期管理——这些地方错一点,性能就从 10k QPS 掉到几百。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

389

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

389

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

571

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

undefined是什么
undefined是什么

undefined是代表一个值或变量不存在或未定义的状态。它可以作为默认值来判断一个变量是否已经被赋值,也可以用于设置默认参数值。尽管在不同的编程语言中,undefined可能具有不同的含义和用法,但理解undefined的概念可以帮助我们更好地理解和编写程序。本专题为大家提供undefined相关的各种文章、以及下载和课程。

4692

2023.07.31

网页undefined是什么意思
网页undefined是什么意思

网页undefined是指页面出现了未知错误的意思,提示undefined一般是在开发网站的时候定义不正确或是转换不正确,或是找不到定义才会提示undefined未定义这个错误。想了解更多的相关内容,可以阅读本专题下面的文章。

2952

2024.08.14

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

2

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.8万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

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

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