0

0

C#文件流与管道(Pipe) C#如何将文件流连接到System.IO.Pipelines

月夜之吻

月夜之吻

发布时间:2026-03-03 09:02:52

|

300人浏览过

|

来源于php中文网

原创

必须手动分块读取 filestream 并写入 pipewriter:用 readasync + getmemory + advance + flushasync 协同控制背压,读完后调 flushasync 再 completeasync;禁用 copytoasync 和 streampipereader,避免同步 i/o 卡死和内存管理失效。

c#文件流与管道(pipe) c#如何将文件流连接到system.io.pipelines

怎么把 FileStream 接到 PipeWriter 上

不能直接塞,PipeWriter 不接受 Stream,它只认 ReadOnlyMemory<byte></byte> 或异步写入操作。你得自己读文件、分块送进去,再手动处理背压和完成信号。

常见错误是用 stream.CopyToAsync(pipe.Writer.AsStream()) —— 这会绕过管道的内存管理,丢失节流能力,还可能触发 InvalidOperationException:“PipeWriter has been completed.”

  • Pipe 自带的 Writer,别转成 Stream 再套一层
  • 读文件用 FileStream.ReadAsync 配合栈分配的 Memory<byte></byte>(比如 stackalloc byte[8192]),避免 GC 压力
  • 每次 WriteAsync 后必须调用 FlushAsync,否则数据卡在内存里不出去
  • 文件读完后要 await writer.CompleteAsync(),否则下游永远等不到 IsCompleted 为 true

为什么不能用 StreamPipeReader 包装 FileStream

StreamPipeReader 是为“可随时读取的流”设计的,比如网络流或内存流;而 FileStream 在某些场景下(如打开时没加 FileOptions.Asynchronous)会退化为同步 I/O,导致 StreamPipeReader 内部线程池卡死,出现高延迟或 TimeoutException

Mokker AI
Mokker AI

AI产品图添加背景

下载
  • Windows 上默认创建的 FileStream 若没显式传 FileOptions.Asynchronous,即使调 ReadAsync 也可能是同步完成的
  • StreamPipeReader 依赖底层流真正异步,否则它的内部缓冲策略会失效
  • 实测中,大文件传输时吞吐量可能比手写循环低 30% 以上,因为多了一层拷贝 + 锁竞争

如何安全地边读文件边写 PipeWriter(含背压)

核心是把文件读取和管道写入做成一个协同节奏:读一块、等写完、再读下一块。靠 PipeWriter.GetMemory()Advance() 控制内存生命周期,而不是预分配大 buffer。

var buffer = writer.GetMemory(8192);
var read = await fileStream.ReadAsync(buffer, cancellationToken);
if (read == 0) break;
writer.Advance(read);
await writer.FlushAsync(cancellationToken);
  • 不要假设 GetMemory(n) 一定返回 n 字节 —— 它可能更少,尤其管道快满时,得检查 .Length
  • 每次 Advance() 后必须 FlushAsync(),否则 GetMemory() 下次可能返回 0 长度
  • 如果文件很大,建议在循环里加 cancellationToken.ThrowIfCancellationRequested(),否则中断时 CompleteAsync 可能被跳过

PipeReader 消费端收不到 EOF?检查 Complete 调用时机

很多问题不是流没读完,而是 writer.CompleteAsync() 调早了:比如在 ReadAsync 返回 0 之后立刻 Complete,但上一批数据还没 Flush 完,导致下游 reader.ReadAsync() 先拿到数据、再等半天才收到 IsCompleted

  • 正确顺序是:ReadAsync → if (0) { await writer.FlushAsync(); await writer.CompleteAsync(); }
  • 如果用了 try/finally 做清理,确保 CompleteAsyncFlushAsync 之后,且不被异常跳过
  • 下游用 while (!reader.IsCompleted) 循环时,记得每次 ReadResult 后调 reader.AdvanceTo(),否则管道内存不释放,很快 OOM
实际跑起来最易忽略的是 FlushAsync 的必要性 —— 它不像 Stream.Flush() 那样可选,而是管道数据流动的开关。漏掉一次,整个链路就卡住,还很难定位。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

839

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

104

2023.09.25

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

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

432

2023.07.18

堆和栈区别
堆和栈区别

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

600

2023.08.10

length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

952

2023.09.19

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

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

723

2023.08.10

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

1379

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1159

2023.07.27

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

3

2026.03.03

热门下载

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

精品课程

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

共94课时 | 10.6万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 20.2万人学习

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

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