使用BlockingCollection更新ObservableCollection

kst*_*s51 5 c# wpf observablecollection blockingcollection

我订阅了一项服务,该服务将在收到新元素时引发Event,然后将此元素添加到中BlockingCollection
我正在运行第二个线程,该线程将循环运行BlockingCollection以添加/更新可观察集合中的元素。

问题是您如何添加ObservableCollection?我知道我不能只.add对这种类型的集合执行操作,因为它需要在UI线程上完成。所以我尝试使用不同的ObservableCollection子类,该子类使用分派器来编组元素的添加,但是每一次,我都会遇到相同的错误

“未知模块中发生了'System.StackOverflowException类型的未处理的异常。”

故障排除提示如下:

确保您没有无限循环或无限递归。

好吧,实际上,我确实有某种无限循环,因为一直在BlockingQueue接收新元素,例如每秒5到10。
如果我不将控件绑定到observablecollectionough,或者使用List代替,则不会出现异常。

Class ElementHolder
{
    private ExternalService _externalService;
    private ObservableCollection<Element> _elementsList = new ObservableCollection<Element>();
    private BlockingCollection<Element> _elementsReceived = new BlockingCollection<Element>();

    public ObservableCollection<Element> ElementsList
    {
        get
        {
            return _elementList;
        }
        set
        {
            _elementList = value;
        }
    }

public ElementHolder()
    {
        Task.Factory.StartNew(ReadElements);
        _externalService = new ExternalService();
        _externalService.ReceivedNewElement += new Action<Element>(o => _elementsReceived.Add(o));
        _externalService.Subscribe();
    }

private void ReadElements()
    {
        foreach (Element element in _elementsReceived.GetConsumingEnumerable())
        {
            Element item = _elementsList.FirstOrDefault(o => o.ID == element.ID);
            if (item == null)
            {
                _elementList.Add(element);
            }
            else
            {
                item.Update(element);
            }
        }
    }
Run Code Online (Sandbox Code Playgroud)

编辑当我跟踪它时,该错误自行消失了。我试图使事情变得更简单,以真正了解问题所在,然后开始起作用。当将所有内容放回原处时,它仍然可以工作……但是它有时会返回,原因似乎并不相关,例如在我的列表视图中添加样式。我开始认为第三方dll中存在问题。

the*_*yer 3

这是一个完美的例子,说明反应式扩展是一个非常有用和巧妙的工具。使用它们有一个相当陡峭的学习曲线,但由于您在这里有一个特定的案例,假设我正确理解您的目标,我将插入将实现您的目标的反应式代码。

请注意,您需要安装 Reactive Extensions,并且需要两个 using 语句(System.Reactive.Linq 和 System.Reactive.Subjects),并且需要引用 System.Reactive 和 System.Reactive.Windows.Threading dll。请参阅MSDN 上的响应式

class ElementHolder
{
    public ObservableCollection<Element> ElementsList { get; set; }
    private ExternalService _externalService = new ExternalService();
    private IDisposable _elementSubscription;
    private Subject<Element> _elementSubject = new Subject<Element>();

    public ElementHolder()
    {
        _externalService.ReceivedNewElement += _elementSubject.OnNext;
        _externalService.Subscribe();

        ElementList = new ObservableCollection<Element>();
        _elementSubscription = _externalService.ObserveOnDispatcher().Subscribe(NextElement);
    }

    private void NextElement(Element e)
    {
        Element item = ElementsList.FirstOrDefault(o => o.ID == element.ID);
        if (item == null) {
            _elementList.Add(element);
        }
        else {
            item.Update(element);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)