如何使ObservableCollection线程安全?

问题描述 投票:15回答:4
System.InvalidOperationException: Collection was modified; enumeration operation may not execute.

我正在添加/删除不在UI线程上的ObservableCollection。

我有一个方法名称Enqueue Report要添加到集合中,还有一个Dequeue Report要从colleciton中删除。

步骤流程如下: -

  1. 1.call EnqueueReport每当请求新报告时
  2. 每隔几秒调用一个方法来检查是否生成了报告(这有一个foreach循环,用于检查ObservableCollection中所有报告的生成状态)
  3. 如果生成报告,则调用DequeueReport

我在C#库中并不多。有人可以指导我吗?

c# thread-safety xamarin observablecollection
4个回答
6
投票

您可以创建一个简单的线程友好版本的observable集合。如下:

 public class MTObservableCollection<T> : ObservableCollection<T>
    {
        public override event NotifyCollectionChangedEventHandler CollectionChanged;
        protected override void OnCollectionChanged(NotifyCollectionChangedEventArgs e)
        {
            NotifyCollectionChangedEventHandler CollectionChanged = this.CollectionChanged;
            if (CollectionChanged != null)
                foreach (NotifyCollectionChangedEventHandler nh in CollectionChanged.GetInvocationList())
                {
                    DispatcherObject dispObj = nh.Target as DispatcherObject;
                    if (dispObj != null)
                    {
                        Dispatcher dispatcher = dispObj.Dispatcher;
                        if (dispatcher != null && !dispatcher.CheckAccess())
                        {
                            dispatcher.BeginInvoke(
                                (Action)(() => nh.Invoke(this,
                                    new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset))),
                                DispatcherPriority.DataBind);
                            continue;
                        }
                    }
                    nh.Invoke(this, e);
                }
        }
    }

现在做一个大规模的发现和替换,并将你所有的ObservableCollection改为MTObservableCollection和你的好去


25
投票

从.net framework 4.5开始,您可以使用本机集合同步。

BindingOperations.EnableCollectionSynchronization(YourCollection, YourLockObject);

YourLockObject是任何对象的实例,例如new Object();。每个系列使用一个。

这消除了一些特殊类或任何东西的需要。只需启用并享受;)

[编辑]正如Mark和Ed的评论中所述(感谢澄清!),这并不能减轻您锁定集合的更新,因为它只是同步集合视图绑定并且不会神奇地使集合线程安全本身。 [/编辑]

PS:BindingOperations居住在Namespace System.Windows.Data


18
投票

Franck在这里发布的解决方案将适用于一个线程正在添加内容的情况,但ObservableCollection本身(以及它所基于的List)不是线程安全的。如果多个线程正在写入集合,则可能会引入难以跟踪的错误。我编写了一个ObservableCollection版本,它使用ReaderWriteLockSlim来实现真正的线程安全。

Unfortunately, it hit the StackOverflow character limit, so here it is on PasteBin.这应该100%与多个读者/作者一起工作。就像常规的ObservableCollection一样,在回调中修改集合(在收到回调的线程上)是无效的。


10
投票

您可以使用Observable Concurrent Collection类。它们位于Microsoft在Parallel Extensions Extras库中提供的包中。

您可以在Nuget:https://www.nuget.org/packages/ParallelExtensionsExtras/上由社区预建

或者从Microsoft获取它:

https://code.msdn.microsoft.com/ParExtSamples

© www.soinside.com 2019 - 2024. All rights reserved.