c# – 使用asyncawait的最佳实践

c# – 使用asyncawait的最佳实践,第1张

概述假设我有以下类定义: public class Calculator{ public CalculatorResult Calculate() { return LongRunningCalculation(); } private CalculatorResult LongRunningCalculation() { r 假设我有以下类定义:
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层)和业务逻辑之外.

总结

以上是内存溢出为你收集整理的c# – 使用async / await的最佳实践全部内容,希望文章能够帮你解决c# – 使用async / await的最佳实践所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/langs/1262971.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2022-06-08
下一篇2022-06-08

发表评论

登录后才能评论

评论列表(0条)

    保存