0

0

C# gRPC流式处理方法 C#如何实现服务器和客户端流

月夜之吻

月夜之吻

发布时间:2026-01-28 13:13:02

|

188人浏览过

|

来源于php中文网

原创

双向流式 RPC是gRPC中唯一支持客户端与服务器同时持续双向通信的模式,双方各自维护独立读写通道,可随时异步读写且互不阻塞,需在.proto中用两个stream关键字定义,并由应用层自行约定消息协议。

c# grpc流式处理方法 c#如何实现服务器和客户端流

什么是 gRPC 的双向流式 RPC?

gRPC 的 BidirectionalStreamingRpc 是唯一支持服务器和客户端同时持续发消息的调用模式。它不是“先发后收”或“发一次收多次”,而是双方各自维护独立的读写通道,可随时 WriteAsync、随时 ReadAsync,彼此不阻塞。

常见误用是把它当“增强版客户端流”或“带回传的服务器流”——实际它没有隐含顺序约定,应用层必须自行定义协议(比如用 messageType 字段区分心跳、数据、结束信号)。

定义 .proto 时必须用 stream 修饰符两次

服务端流、客户端流、双向流在 .proto 中全靠 stream 出现次数区分,少一个关键字就生成完全不同的 C# 方法签名。

  • rpc Chat(stream ChatMessage) returns (ChatMessage) → 客户端流(IAsyncEnumerable 入,单次返回)
  • rpc Chat(ChatMessage) returns (stream ChatMessage) → 服务器流(单次入,IAsyncStreamReader 出)
  • rpc Chat(stream ChatMessage) returns (stream ChatMessage) → 双向流(两个 stream)→ 生成 Task Chat(IAsyncStreamReader, IServerStreamWriter, ServerCallContext)

如果生成的 C# 类里没看到 IServerStreamWriter 参数,八成是 .proto 少写了一个 stream

客户端发起双向流:别卡在 await foreach 里等响应

典型错误是写成:

var call = client.Chat();
await call.RequestStream.WriteAsync(new ChatMessage { Text = "hi" });
await foreach (var msg in call.ResponseStream.ReadAllAsync()) { /* ... */ } // ❌ 阻塞在此,后续 WriteAsync 不会执行

正确做法是并发驱动读写:

RecoveryFox AI
RecoveryFox AI

AI驱动的数据恢复、文件恢复工具

下载
  • Task.Run(() => WriteLoop(call.RequestStream)) 单独发消息
  • await foreach 在主线程收消息
  • 或用 Channel + Writer/Reader 解耦生产和消费

注意 RequestStream.CompleteAsync() 必须显式调用,否则服务端 ReadAsync() 永远不会返回 null

服务端处理双向流:避免在循环里 await 所有操作

最易被忽略的是线程调度开销。下面这段代码在高并发下会迅速堆积 Task

while (await requestStream.ReadAsync(out var req)) {
    var resp = Process(req);
    await responseStream.WriteAsync(resp); // ❌ 每次 WriteAsync 都可能跨线程调度
}

优化方向:

  • 批量写入:缓存多个响应,用 WriteBatchAsync(需自实现或用 Channel 聚合)
  • 取消检查:在循环头部加 context.CancellationToken.ThrowIfCancellationRequested(),否则客户端断连后服务端仍空转
  • 异常隔离:ReadAsync 抛异常时,WriteAsync 可能已失败,不要假设响应一定送达

真正难调试的不是连接断开,而是某次 WriteAsync 因网络抖动超时后,后续所有写入都静默失败——因为 gRPC 的 IServerStreamWriter 不抛异常,只记录日志。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

458

2024.03.01

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

74

2025.12.04

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

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

396

2023.07.18

堆和栈区别
堆和栈区别

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

575

2023.08.10

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

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

503

2023.08.10

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

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

503

2023.08.10

Golang channel原理
Golang channel原理

本专题整合了Golang channel通信相关介绍,阅读专题下面的文章了解更多详细内容。

248

2025.11.14

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

158

2026.01.28

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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