在我的 MVVM 应用程序中,我的视图模型调用 3 个不同的服务方法,将每个方法的数据转换为通用格式,然后使用属性通知/可观察集合等更新 UI。
服务层中的每个方法都会启动一个新方法Task
并将其返回Task
给视图模型。这是我的一种服务方法的示例。
public class ResourceService
{
internal static Task LoadResources(Action<IEnumerable<Resource>> completedCallback, Action<Exception> errorCallback)
{
var t = Task.Factory.StartNew(() =>
{
//... get resources from somewhere
return resources;
});
t.ContinueWith(task =>
{
if (task.IsFaulted)
{
errorCallback(task.Exception);
return;
}
completedCallback(task.Result);
}, TaskScheduler.FromCurrentSynchronizationContext());
return t;
}
}
这是视图模型的调用代码和其他相关部分......
private ObservableCollection<DataItem> Data = new ObservableCollection<DataItem>();
public ICollectionView DataView
{
get { return _dataView; }
set
{
if (_dataView != value)
{
_dataView = value;
RaisePropertyChange(() => DataView);
}
}
}
private void LoadData()
{
SetBusy("Loading...");
Data.Clear();
Task[] tasks = new Task[3]
{
LoadTools(),
LoadResources(),
LoadPersonel()
};
Task.WaitAll(tasks);
DataView = CollectionViewSource.GetDefaultView(Data);
DataView.Filter = FilterTimelineData;
IsBusy = false;
}
private Task LoadResources()
{
return ResourceService.LoadResources(resources =>
{
foreach(var r in resources)
{
var d = convertResource(r);
Data.Add(d);
}
},
error =>
{
// do some error handling
});
}
这几乎可行,但有几个小问题。
数字 1:在一开始的调用中SetBusy
,在我开始任何任务之前和调用之前WaitAll
,我将IsBusy
属性设置为 true。这应该会更新 UI 并显示 BusyIndicator 控件,但它不起作用。我还尝试添加简单的字符串属性并绑定它们,它们也没有被更新。IsBusy 功能是基类的一部分,适用于其他视图模型,在这些视图模型中,我没有运行多个任务,因此我认为 XAML 中的属性通知或数据绑定没有问题。
在整个方法完成后,所有数据绑定似乎都会更新。我在输出窗口中没有看到任何“第一次异常”或绑定错误,这使我相信 UI 线程在调用 WaitAll 之前以某种方式被阻塞。
2 号:我似乎从服务方法返回了错误的任务。我希望WaitAll
在视图模型转换回调中所有服务方法的所有结果之后运行所有内容。但是,如果我从服务方法返回延续任务,则延续永远不会被调用并WaitAll
永远等待。奇怪的是绑定到 ICollectionView 的 UI 控件实际上正确显示了所有内容,我认为这是因为 Data 是一个可观察的集合,而 CollectionViewSource 知道集合更改的事件。