调用线程无法访问此对象,因为另一个线程拥有它 - BackgroundWorker错误

时间:2014-12-25 10:25:02

标签: c# wpf multithreading mvvm backgroundworker

我有下面的代码示例。我想使用BackGroundWorker但是我有一个错误。

让我解释一下代码;

  • GridData属性是视图上DataGrid的源。
  • 我有一个将Excel数据保存到db的按钮,它与SaveExcel:ICommand类绑定。
  • 当我按下此按钮时,它会在viewmodel中调用SaveExcel方法。
  • 保存excel后的SaveExcel方法中我想刷新网格数据,所以我用数据表设置GridData属性。
  • 当我设置此属性时,将调用PropertyChanged事件,该事件在SaveGrid:ICommand类中委派。(此类与另一个按钮绑定)
  • 这里,CanExecuteChanged方法给我错误,“调用线程无法访问此对象,因为不同的线程拥有它”。

我该如何解决这个问题?

任何帮助表示感谢。

            public class MainViewModel
            {
                public DataTable GridData{get;set;}

                public void SaveExcel()
                {
                    .
                    .
                    .
                    RefreshGridData();
                }

                public void RefreshGridData()
                {
                    .
                    .
                    .
                    GridData = <selectedGridData>;
                }

                private void bgw_DoWork(object sender, DoWorkEventArgs e)
                {
                    SaveExcel();
                }
            }


            public class SaveExcel : ICommand
            {
                private MainViewModel viewModel;
                public SaveExcel(MainViewModel viewModel)
                {
                    this.viewModel = viewModel;
                    viewModel.PropertyChanged += (s, e) =>
                    {
                        if (CanExecuteChanged != null &&
                            (e.PropertyName == "SelectedA" || e.PropertyName == "SelectedB"))
                        {
                            CanExecuteChanged(this, new EventArgs());
                        }
                    };
                }

                public bool CanExecute(object parameter)
                {
                    return (viewModel.SelectedA != null && viewModel.SelectedB != null);
                }

                public event EventHandler CanExecuteChanged;

                public void Execute(object parameter)
                {
                    viewModel.bgw.RunWorkerAsync();
                }
            }


            public class SaveGrid : ICommand
            {
                private MainViewModel viewModel;
                public SaveGrid(MainViewModel viewModel)
                { 
                    this.viewModel = viewModel;
                    viewModel.PropertyChanged += (s, e) =>
                    {
                        if (CanExecuteChanged != null && e.PropertyName == "GridData")
                        {
                            CanExecuteChanged(this, new EventArgs());
                        }
                    };
                }

                public bool CanExecute(object parameter)
                {
                    return (viewModel.GridData.Rows.Count > 0);
                }

                public event EventHandler CanExecuteChanged;

                .
                .
                .
                .

            }

1 个答案:

答案 0 :(得分:4)

您需要在UI线程上调用刷新,在WPF中,您可以通过在代码中的适当位置调用Dispatcher.Invoke来执行此操作。 E.g

Application.Current.Dispatcher.BeginInvoke(DispatcherPriority.Background, new Action(() => RefreshGridData()));