gpt4 book ai didi

c# - 提高流式传输大(1-10 GB)文件的速度 .Net Core

转载 作者:行者123 更新时间:2023-12-04 10:22:44 25 4
gpt4 key购买 nike

我正在尝试使用 multipartform-data 通过我的 API 上传 *.iso 文件并将它们流式传输到本地文件夹中。
我使用了 Stream.CopyAsync(destinationStream) 并且它运行缓慢,但还不错。但现在我需要报告进展。所以我使用了自定义 CopyTOAsync 并为其添加了进度报告。但是该方法非常慢(根本 Not Acceptable ),即使与 Stream::CopyToASync 相比也是如此。

 public async Task CopyToAsync(Stream source, Stream destination, long? contentLength, ICommandContext context, int bufferSize = 81920 )
{
var buffer = new byte[bufferSize];
int bytesRead;
long totalRead = 0;
while ((bytesRead = await source.ReadAsync(buffer, 0, buffer.Length)) > 0)
{
await destination.WriteAsync(buffer, 0, bytesRead);
totalRead += bytesRead;
context.Report(CommandResources.RID_IMAGE_LOADIND, Math.Clamp((uint)((totalRead * 100) / contentLength), 3, 99));
}
_logger.Info($"Total read during upload : {totalRead}");
}

我试过的:
Stream::CopyToAsync 的默认缓冲区大小是 81920 字节,我首先使用相同的值,然后我尝试将缓冲区大小增加到 104857600 字节 - 没有区别。

您对如何提高自定义 CopyToAsync 的性能还有其他想法吗?

最佳答案

  • 始终使用 ConfigureAwait await为异步延续指定线程同步。
  • 视平台而定,省略 ConfigureAwait可能默认与 UI 线程(WPF、WinForms)或任何线程(ASP.NET Core)同步。如果它与您的 Stream 复制操作中的 UI 线程同步,那么性能会急剧下降也就不足为奇了。
  • 如果您在线程同步上下文中运行代码,那么您的 await语句将被不必要地延迟,因为程序将继续安排到一个可能很忙的线程。
  • 使用大小至少为几百 KiB 的缓冲区 - 甚至是兆字节大小的缓冲区用于异步操作 - 而不是典型的 4KiB 或 80KiB 大小的数组。
  • This QA shows benchmarks that demonstrate that significantly larger buffers are necessary for async IO to have better performance than synchronous IO .
  • 如果您使用的是 FileStream确保您使用过 FileOptions.AsynchronoususeAsync: true否则 FileStream将通过使用线程池线程而不是 Windows 的 native 异步 IO 执行阻塞 IO 来伪造其异步操作。

  • 关于您的实际代码 - 只需使用 Stream::CopyToAsync而不是自己重新实现它。如果您想要进度报告,请考虑子类化 Stream (作为代理包装器)代替。

    以下是我将如何编写您的代码:
  • First, add my ProxyStream class from this GitHub Gist到您的项目。
  • 然后子类 ProxyStream添加对 IProgress 的支持:
  • 确保任何 FileStream实例是用 FileOptions.Asynchronous | FileOptions.SequentialScan 创建的.
  • 使用 CopyToAsync .
  • public class ProgressProxyStream : ProxyStream
    {
    private readonly IProgress<(Int64 soFar, Int64? total)> progress;
    private readonly Int64? total;

    public ProgressProxyStream( Stream stream, IProgress<Int64> progress, Boolean leaveOpen )
    : base( stream, leaveOpen )
    {
    this.progress = progress ?? throw new ArgumentNullException(nameof(progress));
    this.total = stream.CanSeek ? stream.Length : (Int64?)null;
    }

    public override Task<Int32> ReadAsync( Byte[] buffer, Int32 offset, Int32 count, CancellationToken cancellationToken )
    {
    this.progress.Report( ( offset, this.total ) );
    return this.Stream.ReadAsync( buffer, offset, count, cancellationToken );
    }
    }


    如果性能仍然受到上述影响 ProgressProxyStream那么我敢打赌瓶颈在 IProgress.Report里面回调目标(我假设它与 UI 线程同步)- in which case a better solution is to use a ( System.Threading.Channels.Channel )ProgressProxyStream (或什至您对 IProgress<T> 的实现)将进度报告转储到不阻塞任何其他 IO 事件。

    关于c# - 提高流式传输大(1-10 GB)文件的速度 .Net Core,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60772727/

    25 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com