从基于事件的异步迁移到基于任务的异步

时间:2013-10-10 18:27:48

标签: c# wpf asynchronous

我正在使用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订购它。

整个任务概念对我来说相当新,所以也许我错过了一些东西。有没有比我上面列出的更简洁的方式?

3 个答案:

答案 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是返回的数据,而不是让方法设置其他字段。