- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在这样的后台线程中为 wpf 窗口获取数据 [framework 4.0 with async/await]:
async void refresh()
{
// returns object of type Instances
DataContext = await Task.Factory.StartNew(() => serviceagent.GetInstances());
var instances = DataContext as Instances;
await Task.Factory.StartNew(() => serviceagent.GetGroups(instances));
// * problem here * instances.Groups is filled but UI not updated
}
当我在 GetInstances 中包含 GetGroups 的操作时,UI 会显示组。
当我在单独的操作中更新时,DataContext 包括正确的组,但 UI 不显示它们。
在 GetGroups()
我包含的方法NotifyCollectionChangedAction.Reset
对于ObservableCollection
组,这没有帮助。
更奇怪的是我调用NotifyCollectionChangedAction.Reset
列表中只有一次,但执行了 3 次,而列表有 10 项?!
我可以通过写来解决这个问题:
DataContext = await Task.Factory.StartNew(() => serviceagent.GetGroups(instances));
但这是通过后台进程更新 DataContxt 和 UI 的常规方式吗?
其实我只想更新现有的DataContext而不重新设置?
编辑:serviceagent.GetGroups(instances)
更详细:
public void GetGroups(Instances instances)
{
// web call
instances.Admin = service.GetAdmin();
// set groups for binding in UI
instances.Groups = new ViewModelCollection<Groep>(instances.Admin.Groups);
// this code has no effect
instances.Groups.RaiseCollectionChanged();
}
这里 ViewModelCollection<T>
继承自 ObservableCollection<T>
我添加了方法:
public void RaiseCollectionChanged()
{
var handler = CollectionChanged;
if (handler != null)
{
Trace.WriteLine("collection changed");
var e = new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset);
handler(this, e);
}
}
最佳答案
您的代码的 async
部分有几点很突出:
async void
在我的 MSDN 文章中。总之,对于 async
方法来说,void
是一种不自然的返回类型,因此它有一些怪癖,尤其是在异常处理方面。TaskEx.Run
over StartNew
for asynchronous tasks ,正如我在博客中解释的那样。基于这些,我也推荐我的intro to async
博文。
关于实际问题...
从后台线程更新数据绑定(bind)代码总是很棘手。我建议您将 ViewModel 数据视为 UI 的一部分(可以说是“逻辑 UI”)。所以可以在后台线程上检索数据,但更新实际 VM 值应该在 UI 线程上完成。
这些更改使您的代码看起来更像这样:
async Task RefreshAsync()
{
var instances = await TaskEx.Run(() => serviceagent.GetInstances());
DataContext = instances;
var groupResults = await TaskEx.Run(() => serviceagent.GetGroups(instances));
instances.Admin = groupResults.Admin;
instances.Groups = new ObservableCollection<Group>(groupResults.Groups);
}
public GroupsResult GetGroups(Instances instances)
{
return new GroupsResult
{
Admin = service.GetAdmin(),
Groups = Admin.Groups.ToArray(),
};
}
接下来需要检查的是Instances
是否实现了INotifyPropertyChanged
。设置 Groups
时不需要引发 Reset
集合更改事件;由于 Groups
是 Instances
上的一个属性,因此 Instances
有责任引发 INotifyPropertyChanged.PropertyChanged
。
或者,您可以将 DataContext
设置为最后:
async Task RefreshAsync()
{
var instances = await TaskEx.Run(() => serviceagent.GetInstances());
var groupResults = await TaskEx.Run(() => serviceagent.GetGroups(instances));
instances.Admin = groupResults.Admin;
instances.Groups = new ObservableCollection<Group>(groupResults.Admin.Groups);
DataContext = instances;
}
关于wpf - 从后台线程更新 DataContext,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19903306/
我是一名优秀的程序员,十分优秀!