在视图模型中,我使用工厂:
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)
中看到这一点:由于嵌套等待,请参阅注释。
答案 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();
};