WPF MVVMLight:在ViewModel中,开子线程为ObservableCollectio

2021-02-01 16:48发布

问题:

我在项目中使用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>