2

由于如此强调基于任务的异步开发,我惊讶地发现 Windows Phone 8 中的服务引用不支持基于任务的实现。相反,他们使用“XYZCompleted”基于事件的异步方法。

因此,需要这样的代码:http: //codepaste.net/fqyt47

public async Task<IEnumerable<MyService.Character>> GetCharactersAsync()
{
    var _Task = new TaskCompletionSource<IEnumerable<MyService.Character>>();
    var _Client = new MyService.ServiceClient();
    _Client.GetCharactersCompleted += (s, e) =>
    {
        var _Characters = e.Result as IEnumerable<MyService.Character>;
        if (e.Error != null && !_Task.TrySetException(e.Error))
            System.Diagnostics.Debugger.Break();
        else if (e.Cancelled && !_Task.TrySetCanceled())
            System.Diagnostics.Debugger.Break();
        else if (!_Task.TrySetResult(_Characters))
            System.Diagnostics.Debugger.Break();
    };
    _Client.GetCharactersAsync();

    return await _Task.Task;
}

然而,这感觉不对。有没有更好、更优雅的方法?

4

1 回答 1

1

我很惊讶 Windows Phone 8 没有Task基于 - 的参考创建。

也就是说,TaskCompletionSource 是与各种异步模式(包括 EAP)进行互操作的标准方式

通常它是通过扩展方法完成的,这些扩展方法可以被消耗await但它们本身不是async

public Task<IEnumerable<MyService.Character>> GetCharactersTaskAsync(this ServiceClient client)
{
  var tcs = new TaskCompletionSource<IEnumerable<MyService.Character>>();
  client.GetCharactersCompleted += (s, e) =>
  {
    if (e.Error != null) tcs.SetException(e.Error);
    else if (e.Cancelled) tcs.SetCanceled();
    else tcs.SetResult(e.Result);
  };
  client.GetCharactersAsync();
  return tcs.Task;
}
于 2012-11-27T02:28:20.563 回答