Http.IMiddleware不要在未通过TaskScheduler的情况下创建任务

问题描述 投票:-1回答:1

(。Net 3.1,Visual Studio 2019)

在DevExpress代码示例:https://github.com/DevExpress-Examples/blazor-server-dxdatagrid-export/blob/19.2.2%2B/CS/DxDataGridExportingWithReports/Helpers/ExportMiddleware.cs中,以下http中间件代码得到了警告

不要在不通过TaskScheduler的情况下创建任务...

重写代码以启动新任务的正确方法是什么?

public class ExportMiddleware : IMiddleware
{
    ......

    public Task InvokeAsync(HttpContext context, RequestDelegate next)
    {
       ......

        _ = await new TaskFactory().StartNew(() => // warning: Do not create tasks without passing a TaskScheduler
        {
            report.CreateDocument();
            using (MemoryStream fs = new MemoryStream())
            {
                if (format == pdf)
                    report.ExportToPdf(fs);
                else if (format == xlsx)
                    report.ExportToXlsx(fs);
                else if (format == docx)
                    report.ExportToDocx(fs);
                context.Response.Clear();
                context.Response.Headers.Append("Content-Type", "application/" + format);
                context.Response.Headers.Append("Content-Transfer-Encoding", "binary");
                context.Response.Headers.Append("Content-Disposition", "attachment; filename=ExportedDocument." + format);
                context.Response.Body.WriteAsync(fs.ToArray(), 0, fs.ToArray().Length);
                return context.Response.CompleteAsync();
            }
        });
c# asp.net asp.net-core devexpress task-parallel-library
1个回答
0
投票

技术上,为了回答实际问题,代码应使用Task.Run而不是StartNew

_ = Task.Run(() =>

但是,这是一个非常糟糕的主意。这不仅会执行fire-and-forget,而且将来还会在某个随机时间使用context.Response。然后它将开始对响应流进行写操作,然后在写操作完成之前完成响应流。它完全坏了。

相信一个更合适的解决方案是完全删除工厂/启动/运行并在必要时使用await

public async Task InvokeAsync(HttpContext context, RequestDelegate next)
{
  ......
  report.CreateDocument();
  using (MemoryStream fs = new MemoryStream())
  {
    if (format == pdf)
      report.ExportToPdf(fs);
    else if (format == xlsx)
      report.ExportToXlsx(fs);
    else if (format == docx)
      report.ExportToDocx(fs);
    context.Response.Clear();
    context.Response.Headers.Append("Content-Type", "application/" + format);
    context.Response.Headers.Append("Content-Transfer-Encoding", "binary");
    context.Response.Headers.Append("Content-Disposition", "attachment; filename=ExportedDocument." + format);
    await context.Response.Body.WriteAsync(fs.ToArray(), 0, fs.ToArray().Length);
    await context.Response.CompleteAsync();
  }
}
© www.soinside.com 2019 - 2024. All rights reserved.