假设我有以下类定义:
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();
}
}
Run Code Online (Sandbox Code Playgroud)
如果我想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);
}
Run Code Online (Sandbox Code Playgroud)
在我拥有库的情况下,我认为还有两个选项,其中一个与第一个非常相似.可能是由于缺乏我自己的理解.
创建一个on on方法ClassThatUsesACalculator
,封装该DoWork()
方法,然后从表单上的异步方法调用该方法.
要么,
封装LongRunningCalculation()
在Calculator
同一个班级Task.Run()
.
public Task<CalculatorResult> CalculateAsync()
{
return Task.Run(() =>
{
return LongRunningCalculation();
});
}
Run Code Online (Sandbox Code Playgroud)在ClassThatUsesACalculator
等待新创建的方法的调用上创建异步方法.
public async Task DoWorkAsync()
{
for (int i = 0; i < 10; i++)
{
var result = await calculator.CalculateAsync();
DoSomethingWithCalculationResult(result);
DoLightWork();
OnProgressChanged();
}
}
Run Code Online (Sandbox Code Playgroud)在表单上创建一个异步方法(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();
}
Run Code Online (Sandbox Code Playgroud)现在,在我看来,最后一个选项将是最好的,因为我会保持更多的控制权.但也许我已经离开了,希望得到某人的意见或指示,因为我只能找到关于如何消耗异步的解释,但从来没有真正如何构建其他人消费的方法.
Ste*_*ary 11
作为一般规则,Task.Run
尽可能推动调用堆栈的任何使用.
您要避免的是拥有一个带有异步签名的方法,该方法是Task.Run
在可重用组件中实现的.这是一个撒谎的API.我有关于这个主题的博客文章更详细.
如果您控制有问题的类,我建议使用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(...));
}
}
Run Code Online (Sandbox Code Playgroud)
然后使用它非常简单:
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));
}
Run Code Online (Sandbox Code Playgroud)
这样可以在Task.Run
需要它的组件(UI层)和业务逻辑之外使用它.
归档时间: |
|
查看次数: |
490 次 |
最近记录: |