我正在使用WCF服务在WPF应用程序中加载一些数据,直到最近,通过Visual Studio为我自动生成的基于事件的异步方法实现了这一点:
//Old way
private void LoadFoos(int barId)
{
serviceClient.SelectFoosByBarIdCompleted += (s, e) =>
{
Foos = e.Result.OrderBy(f => f.Description).ToList();
});
serviceClient.SelectFoosByBarIdAsync();
}
无论出于何种原因,我们转向使用任务,我对最佳方式做了同样的事情:
//New way
private async void LoadFoos(int barId)
{
private TaskScheduler uiTaskScheduler = TaskScheduler.FromCurrentSynchronizationContext();
serviceClient.SelectFoosByBarIdAsync(barId).ContinueWith(t =>
{
Foos = t.Result.OrderBy(f => f.Description).ToList();
}, uiTaskScheduler);
}
我认为这更加丑陋,因为我必须手动设置上下文,所以我不会在错误的线程上更新内容(Foos
是一个数据绑定属性)。另外,我以为我能够做到这一点:
//New way #2, doesn't sort ;(
private async void LoadFoos(int barId)
{
private TaskScheduler uiTaskScheduler = TaskScheduler.FromCurrentSynchronizationContext();
var selectFoosTask = serviceClient.SelectFoosByBarIdAsync(barId);
Foos = selectFoosTask;
}
但是我不能按照Description
订购它。
整个任务概念对我来说相当新,所以也许我错过了一些东西。有没有比我上面列出的更简洁的方式?
答案 0 :(得分:1)
您只需使用等待,而不是延续:
private async Task LoadFoos(int barId)
{
var temp = await serviceClient.SelectFoosByBarIdAsync(barId);
Foos = temp.OrderBy(f => f.Description).ToList();
}
请注意,使用async void,甚至只使用Task
返回方法可能并不理想。将其重写为(*假设Foos是List<string>
):
private async Task<List<string>> LoadFoosAsync(int barId)
{
var temp = await serviceClient.SelectFoosByBarIdAsync(barId);
return temp.OrderBy(f => f.Description).ToList();
}
然后,当你调用它时,使用:
Foos = await LoadFoosAsync(id);
答案 1 :(得分:1)
至于将基于事件的异步方法转换为基于任务的方法(任务非常优越:))查看这些博客文章
http://msdn.microsoft.com/en-us/magazine/ff959203.aspx http://blogs.msdn.com/b/pfxteam/archive/2009/06/19/9791857.aspx
在编组回ui线程之前,您仍然可以执行所有处理。您还可以创建自己的ContinueWith帮助程序方法,将任务放在正确的TaskScheduler上。 (如果你不能使用等待,这是最简单的选择)
另请注意,wsdl工具的较新版本(2012年及以后我认为)实际上会为服务生成基于任务的异步方法。
答案 2 :(得分:0)
由于该方法只是async
await
任务,而不是手动添加延续:
private async Task LoadFoos(int barId)
{
Foos = (await serviceClient.SelectFoosByBarIdAsync(barId))
.OrderBy(f => f.Description).ToList();
}
另请注意,应尽可能避免使用async void
方法,因为您无法知道异步操作何时结束,或者无法访问任何抛出的异常。而是让方法返回Task
。更好的方法是让方法返回Task<T>
,其中T
是返回的数据,而不是让方法设置其他字段。