c# – 使用async / await的最佳实践

假设我有以下类定义
public class Calculator
{
    public CalculatorResult Calculate()
    {
        return LongRunningCalculation();
    }

    private CalculatorResult LongRunningCalculation()
    {
        return new CalculatorResult(0.00);
    }
}

public class ClassthatUsesACalculator
{
    private readonly Calculator calculator;

    public ClassthatUsesACalculator()
    {
        this.calculator = new Calculator();
    }

    public void DoWork()
    {
        for (int i = 0; i < 10; i++)
        {
            var result = calculator.Calculate();

            DoSomethingWithCalculationResult(result);

            DoLightWork();

            OnProgressChanged();
        }
    }
}

public partial class Form : Form
{
    public Form()
    {
        InitializeComponent();
    }

    private void Method(object sender,EventArgs e)
    {
        DoWork();
    }

    private void DoWork()
    {
        var calculator = new ClassthatUsesACalculator();
        calculator.ProgressChanged += (s,e) =>
        {
            // Update progressbar
        };

        calculator.DoWork();
    }
}

如果我想在DoWork()中完成工作,在表单上,​​异步地我可以添加一个方法(GetCalculationTask),它使用Task.Run()返回一个任务,并添加一个异步事件处理程序,即一个按钮(Methodone).

如果我错了,请纠正我,但在我看来,当ClassthatUsesACalculator和Calculator类驻留在我不拥有的库中时,这将是唯一的选择.

private Task GetCalculationTask(IProgress<CalculatorProgress> progress)
{
    var calculator = new ClassthatUsesACalculator();
    calculator.ProgressChanged += (s,e) =>
    {
        progress.Report(new CalculatorProgress(0));
    };

    return Task.Run(() =>
    {
        calculator.DoWork();
    });
}

private async void Methodone(object sender,EventArgs e)
{
    IProgress<CalculatorProgress> progress = new Progress<CalculatorProgress>   (UpdateProgressBar);

    await GetCalculationTask(progress);
}

在我拥有库的情况下,我认为还有两个选项,其中一个与第一个非常相似.可能是由于缺乏我自己的理解.

在ClassthatUsesACalculator上创建一个封装DoWork()方法方法,然后从表单上的异步方法调用方法.

要么,

>使用Task.Run()封装Calculator类上的LongRunningCalculation().

public Task<CalculatorResult> CalculateAsync()
{
    return Task.Run(() =>
    {
        return LongRunningCalculation();
    });
}

>在ClassthatUsesACalculator上创建等待新创建方法调用异步方法.

public async Task DoWorkAsync()
{
    for (int i = 0; i < 10; i++)
    {
        var result = await calculator.CalculateAsync();

        DoSomethingWithCalculationResult(result);

        DoLightWork();

        OnProgressChanged(); 
    }
}

>在表单上创建一个异步方法(MethodThree)

private async void MethodThree(object sender,EventArgs e)
{
    IProgress<CalculatorProgress> progress = new Progress<CalculatorProgress>(UpdateProgressBar);

    var calculator = new ClassthatUsesACalculator();
    calculator.ProgressChanged += (s,args) =>
    {
        progress.Report(new CalculatorProgress(0));
    };

    await calculator.DoWorkAsync();
}

现在,在我看来,最后一个选项将是最好的,因为我会保持更多的控制权.但也许我已经离开了,希望得到某人的意见或指示,因为我只能找到关于如何消耗异步的解释,但从来没有真正如何构建其他人消费的方法.

解决方法

作为一般规则,尽可能将任何Task.Run用法推送到调用堆栈.

您要避免的是拥有一个带有异步签名的方法,该方法是在可重用组件中使用Task.Run实现的.这是一个撒谎的API.我有一个更详细的blog post on the subject.

如果你控制有问题的类,我建议使用IProgress< T>而不是进展更新的事件. IProgress< T>使用同步代码和异步工作正常:

public void DoWork(IProgress<CalculatorProgress> progress = null)
{
  for (int i = 0; i < 10; i++)
  {
    var result = calculator.Calculate();

    DoSomethingWithCalculationResult(result);

    DoLightWork();

    if (progress != null)
      progress.Report(new CalculatorProgress(...));
  }
}

然后使用它非常简单:

private async void MethodTwo(object sender,EventArgs e)
{
  IProgress<CalculatorProgress> progress = new Progress<CalculatorProgress>(UpdateProgressBar);

  var calculator = new ClassthatUsesACalculator();

  await Task.Run(() => calculator.DoWork(progress));
}

这使得Task.Run的使用保持在需要它的组件(UI层)和业务逻辑之外.

相关文章

在要实现单例模式的类当中添加如下代码:实例化的时候:frmC...
1、如果制作圆角窗体,窗体先继承DOTNETBAR的:public parti...
根据网上资料,自己很粗略的实现了一个winform搜索提示,但是...
近期在做DSOFramer这个控件,打算自己弄一个自定义控件来封装...
今天玩了一把WMI,查询了一下电脑的硬件信息,感觉很多代码都...
最近在研究WinWordControl这个控件,因为上级要求在系统里,...