Async-await似乎使用UI线程

时间:2013-11-18 16:39:57

标签: wpf async-await

在视图模型中,我使用工厂:

private async Task<BaseData> InitializeAsync()
{
    await InstancesAsync();
    await ProjectsAsync();
    await AdminAsync();
    return this;
}
public static async Task<BaseData> CreateAsync()
{
    var ret = new BaseData();
    return await ret.InitializeAsync();
}

等待的方法相当明确,例如。

 var instances = await TaskEx.Run(new Func<List<string>>(() => Agent.GetInstances()));

在wpf视图中,我想在构造函数中设置DataContext:

Loaded += delegate
{
    Dispatcher.Invoke(new Action(async () => { DataContext = await BasisGegevens.CreateAsync(); }));
};

虽然它有效,但我感到相当不舒服,因为UI线程在无处不在,也在等待完成后的回调之后。我错过了什么?

另外我不明白如何使用DataContext的工厂模式,因为如果没有Invoke,我会得到另一个线程拥有该对象的错误。

编辑:使用Cleary先生的想法,我得到:

Loaded += async (object sender, RoutedEventArgs e) =>
          { DataContext = await BaseData.CreateAsync(); };
public static Task<BaseData> CreateAsync()
{
    var ret = new BaseData();
    return ret.InitializeAsync();
}
private async Task<BaseData> InitializeAsync()
{
    // UI thread here
    await InstancesAsync().ConfigureAwait(false);
    // thread 'a' here
    await ProjectsAsync().ConfigureAwait(false);
    // thread 'a' sometimes 'b' here
    await AdminAsync().ConfigureAwait(false);
    // thread 'a' or 'b' here
    return this;
}

这很好,除了我无法理解ConfigureAwait(false)的工作原理 在方法InstancesAsync()内,我有等待的任务:
var instances = await TaskEx.Run(new Func<List<string>>(() => Agent.GetInstances()));
在等待回复之后,我回到UI线程中 - 我从没想过会发生这种情况! 请注意,ProjectsAsync()AdminAsync()的行为相同,尽管它们是在工作线程(或背景线)上开始的! 我认为ConfigureAwait(true)具有在调用线程中返回的效果(在我的情况下是UI线程)。我测试过它就是这样。
为什么我也会在ConfigureAwait(false)中看到这一点:由于嵌套等待,请参阅注释。

1 个答案:

答案 0 :(得分:7)

我认为将ViewModel视为具有UI线程关联性是最有用的。将其视为逻辑UI,即使它不是实际 UI。因此,ViewModel类的所有属性和可观察集合更新都应该在UI线程上完成。

async方法中,如果您不需要返回UI线程,则可以使用ConfigureAwait(false)来避免在UI线程上恢复。例如,如果您的各种初始化方法是独立的,您可以执行以下操作:

private async Task<BaseData> InitializeAsync()
{
  // Start all methods on the UI thread.
  var instancesTask = InstancesAsync();
  var projectsTask = ProjectsAsync();
  var adminTask = AdminAsync();

  // Await for them all to complete, and resume this method on a background thread.
  await Task.WhenAll(instancesTask, projectsTask, adminTask).ConfigureAwait(false);

  return this;
}

此外,只要您return await,请再看看是否可以完全避免async / await

public static Task<BaseData> CreateAsync()
{
  var ret = new BaseData();
  return ret.InitializeAsync();
}

最后,你应该强烈避免Dispatcher。您的Loaded事件可以简化为:

Loaded += async ()
{
  DataContext = await BasisGegevens.CreateAsync();
};