问题:
我在项目中使用TreeView中通过绑定ItemSource为ObservableCollection集合来显示数据。由于构造ObservableCollection的Nodes花费时间比较长,准备使用在子线程中加载,加载完成后把值赋给Nodes。但是报错【 必须在与DependencyObject相同的Thread上创建DependencySource 】
Model代码:
//树视图数据源
private ObservableCollection<Node> nodes = new ObservableCollection<Node> () ;
public ObservableCollection<Node> Nodes
{
get
{
return nodes;
}
set { nodes = value; RaisePropertyChanged(() => Nodes); }
}
填充Nodes代码(临时创建了nodes_c来保存,完成后统一给Nodes,或者完成一个给一个也行):
public void FillNode()
{
try
{
nodes_c.Add(integration_.SocialChat(casePath_, Applist_["测试1"], Userlist_));
nodes_c.Add(integration_.SocialChat(casePath_, Applist_["测试2"], Userlist_));
nodes_c.Add(integration_.SocialChat(casePath_, Applist_["测试3"], Userlist_));
}
catch (Exception ex)
{
//log
}
}
调用填充Nodes方法
public CaseViewModel()
{
//这里是用MvvmLight的Messenger触发的Load方法
Messenger.Default.Register<DataHelper>(this, "DB", Load);
}
public async void Load(DataHelper x)
{
await Task.Run(FillNode);
//这里是网上找的方法,但是不起作用。
//效果等于直接写Nodes = new ObservableCollection<Node>(nodes_c);
ThreadPool.QueueUserWorkItem(delegate
{
System.Threading.SynchronizationContext.SetSynchronizationContext(new
System.Windows.Threading.DispatcherSynchronizationContext(System.Windows.Application.Current.Dispatcher));
System.Threading.SynchronizationContext.Current.Post(pl =>
{
//把FillNode里临时的nodes_c 赋值给XAML绑定的Nodes属性
Nodes = new ObservableCollection<Node>(nodes_c);
LogHelper.log_.Warn(Nodes.Count);
}, null);
});
}
回答1:
用
Application.Current.Dispatcher.Invoke(action);
代替ThreadPool.QueueUserWorkItem的一堆
这个action就是楼上invoke里面的action
用这个直接就invoke到ui线程执行了
回答2:
因为你在异步方法里面用了ui的代码,所以报这个错
回答3:
Invoke(new Action(()=>{
Nodes = new ObservableCollection<Node>(nodes_c);
LogHelper.log_.Warn(Nodes.Count);
}));
回答4:
因为你修改控件的线程不是创建控件的线程。其实你为什么要这样做。你把model绑定到ui,只需修改model,UI自然就变了。你这样做回退到了winform时代了。
回答5:
谢谢各位大佬给的解决方案。实际是我疏忽了代码。我再XAML代码的模板中绑定了Model中的Brush 类型的Color属性,在异步代码中创建Model实例,并且初始化了Color属性。导致跨线程错误。使用转换器解决。
Converter:
public class ColorConverter : IValueConverter
{
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
{
if (UtilityToolSuite.Tool.SafetyTransformInt32(value) == 0)
{
return new SolidColorBrush(Color.FromRgb(0, 0, 0));
}
else
{
return new SolidColorBrush(Color.FromRgb(255, 0, 0));
}
}
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
{
throw new NotImplementedException();
}
}
XAML:
<TextBlock VerticalAlignment="Center">
<Run Foreground="{Binding Del,Converter={StaticResource ColorConverter}}" Text="{Binding Del}" />
</TextBlock>