6

您能否判断以下代码片段是否正确绑定到异步数据源?

虽然它似乎正在工作,即:UI 没有冻结,但我并不完全确定正确性,因为 MSDN 文档并没有真正谈论绑定到这些文档中的“异步”方法:

绑定.IsAsync

ObjectDataProvider.IsAsynchronous

<pages:HottestPageProxy x:Key="PageProxy" ></pages:HottestPageProxy>

<ObjectDataProvider x:Key="DataProviderArtists" IsAsynchronous="True"  ObjectInstance="{StaticResource PageProxy}"  MethodName="GetArtists">
    <ObjectDataProvider.MethodParameters>
        <system:String>Grabbing artists !</system:String>
    </ObjectDataProvider.MethodParameters>
</ObjectDataProvider>

(HottestPageProxy 对象是一个为控件提供数据的小助手)

public class HottestPageProxy
{
    [UsedImplicitly]
    public async Task<ArtistsQuery> GetArtists([CallerMemberName] string memberName = "")
    {
        Console.WriteLine(memberName);
        string apiKey = App.GetApiKey();
        Task<ArtistsQuery> topHottt = Queries.ArtistTopHottt(new ArtistTopHotttParameters
        {
            ApiKey = apiKey,
            Results = 100,
            Buckets = new[] {ArtistTopHotttBuckets.Hotttnesss}
        });
        return (await topHottt);
    }
}

编辑:“等待topHottt”调用的方法

public static async Task<ArtistsQuery> ArtistTopHottt(ArtistTopHotttParameters parameters)
{
    if (parameters == null) throw new ArgumentNullException("parameters");
    return await Get<ArtistsQuery>(parameters);
}

private static async Task<T> Get<T>(Parameters parameters) where T : Query
{
    if (parameters == null) throw new ArgumentNullException("parameters");
    ValidateParameters(parameters);

    string url = parameters.GetQueryUrl();
    var value = new Uri(url);
    using (var client = GetHttpClient())
    using (var message = await client.GetAsync(url))
    {
        // fetch message content (removed)
        return GetQueryResultObject<T>(s);
    }
}

private static T GetQueryResultObject<T>(string json) where T : class
{
    // create T from Json string (removed)
    return t;
}

使用 AsyncEx 进行编辑

使用您的库虽然现在的语法是:

<ItemsControl x:Name="ItemsControlTopHott"
    ItemsSource="{Binding ... Path=Artists.Result.Artists ...}">
</ItemsControl>

'Artists.Result.Artists' 真的是您希望我使用的吗?:)

新语法使其更加混乱,因为来源是:

public sealed class ArtistsQuery : Query
{
    public List<Artist> Artists { get; set; }
}

没什么大不了的,但如果我能避免这样的语法那就太好了。

您说 .Result 可能会带来僵局,那么我在实施您的解决方案时是否遗漏了什么?

Artists.PropertyChanged 事件引发以下消息:

  • 状态: RanToCompletion
  • 已完成:真
  • IsSuccessfullyCompleted: 真
  • 结果:成功

正如您在帖子中提到的那样,我将尝试 .ConfigureAwait(false) 以了解它的效果。

忘了提一下,实际上我使用 .Result 的实现确实不会阻塞 UI,因为获得结果的典型时间是几秒钟;我会看到 UI 冻结。看起来是对的......但我不确定它是否正确,因此我的问题。

4

1 回答 1

7

async正如其他人所指出的,WPF 类型中的“异步”成员与and无关await

您的绑定确实有问题;您的路径是 using Task.Result,这将阻塞 UI 线程,直到查询完成。此外,使用Result会带来我在博客中描述的死锁的可能性。

我有另一个处理async属性的博客条目,特别是如何将数据绑定到(逻辑上)异步的属性。我的AsyncEx 库有一个名为的类型NotifyTaskCompletion,它允许您更自然地将数据绑定到异步Task​​.

因此,例如,您可以执行以下操作:

public class HottestPageProxy
{
  public HottestPageProxy()
  {
    Artists = NotifyTaskCompletion.Create(GetArtists());
  }

  public INotifyTaskCompletion<ArtistsQuery> Artists { get; private set; }

  private Task<ArtistsQuery> GetArtists()
  {
    string apiKey = App.GetApiKey();
    return Queries.ArtistTopHottt(new ArtistTopHotttParameters
    {
        ApiKey = apiKey,
        Results = 100,
        Buckets = new[] {ArtistTopHotttBuckets.Hotttnesss}
    });
  }
}

然后,您可以数据绑定到 上的几个不同属性INotifyTaskCompletion<T>,包括IsCompletedErrorMessageResult不会default(T)阻塞;如果任务未完成,它将返回)。

于 2013-09-10T14:03:15.687 回答