Concurrent observable collection












1














I'm working on a WPF 4.5 desktop application that has several nested collections with the following key aspects:




  • writes mainly from UI thread, but also from worker threads

  • writes are relatively seldom, mainly directly after a user interaction

  • reads from any thread, but esp. one performance critical worker thread with very many reads (iterating in millisecond intervals)

  • with the exception of the thread mentioned above, read/write performance should not be critical

  • items need to be ordered, i.e. an item's position must always stay the same

  • "remove item" must be supported

  • "insert at" must be supported, although I'm aware that indices must be handled with care, if several threads are involved

  • collection will be used as a WPF binding source and must be observable (implement INotifyCollectionChanged and INotifyPropertyChanged so WPF can update the UI, if items are added/removed)


  • collection must support live-shaping (allowing WPF to instantly update a control's sorting/filtering, if relevant items' properties change; requires the underlying collection to implement IList or similar, so a ListCollectionView can be used)

  • a lookup via key is not required (or can be achieved using extension methods, e.g. FirstOrDefault)

  • approx. max. number of collections < 10k

  • approx. max. number of items / collection < 1k


The out-of-the-box system classes have the following issues (for my use-case) which prevent me from using them as-is:





  • System.Collections.Concurrent classes do not implement IList - and cannot be used for live-shaping


  • System.Collections.ObjectModel.ObservableCollection<T> is not thread-safe


So to fulfill all above requirements I created a wrapper class that implements the required interfaces (e.g. IList, INotifyCollectionChanged...). Internally I chose to use List<T>. (I could have chosen ObservableCollection, but I wanted full control when invoking/dispatching CollectionChanged.)



For all write operations the wrapper class uses lock(_lock) and delegates the call to the inner list. Also - from within the lock - it updates an Array snapshot of the current list, stored in a private field, _snapshot. Then - still from within the lock - it uses System.Windows.Threading.Dispatcher.InvokeAsync() to raise the CollectionChanged event on the correct UI thread.



All read operations use the cached _snapshot, esp. GetEnumerator. The intention behind the snapshot is to avoid locking in the GetEnumerator implementation, for performance reasons of the thread with many reads.



Is the approach ok, what am I missing, what else must I be aware of?



Here's my current code (with some omissions), which appears to work:



using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.ComponentModel;
using System.Linq;
using System.Threading;
using System.Windows;

namespace StackOverflow.Questions
{
public class ObservableConcurrentList<T> : IList, IList<T>, INotifyCollectionChanged, INotifyPropertyChanged
{
private readonly System.Windows.Threading.Dispatcher _context;
private readonly IList<T> _list = new List<T>();
private readonly object _lock = new object();
private T _snapshot;

public ObservableConcurrentList()
{
_context = Application.Current?.Dispatcher;

updateSnapshot();

SuppressNotifications = suppressNotifications;
}

public event NotifyCollectionChangedEventHandler CollectionChanged;

public event PropertyChangedEventHandler PropertyChanged;

private void updateSnapshot()
{
lock (_lock) //precautionary; should be re-entry
{
Interlocked.Exchange(ref _snapshot, _list.ToArray());
}
}

private void notify(NotifyCollectionChangedEventArgs args)
{
if (_context == null)
{
invokeCollectionChanged(args);
}
else
{
_context.InvokeAsync(() => invokeCollectionChanged(args));
}
}

private void invokeCollectionChanged(NotifyCollectionChangedEventArgs args)
{
CollectionChanged?.Invoke(this, args);
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Count)));
}

#region IEnumerable
public IEnumerator<T> GetEnumerator()
{
var localSnapshot = _snapshot; //create local variable to protect enumerator, if class member (_snapshot) should be changed/replaced while iterating
return ((IEnumerable<T>)localSnapshot).GetEnumerator();
}

IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
#endregion

#region ICollection<T>
public void Add(T item)
{
lock (_lock)
{
_list.Add(item);
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, _list.Count - 1));
}
}

public bool Contains(T item)
{
return _snapshot.Contains(item);
}

public void CopyTo(T array, int arrayIndex)
{
_snapshot.CopyTo(array, arrayIndex);
}

public bool Remove(T item)
{
lock (_lock)
{
var index = _list.IndexOf(item);
if (index > -1)
{
if (_list.Remove(item))
{
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
return true;
}
}

return false;
}
}

public void Clear()
{
lock (_lock)
{
_list.Clear();
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
}
}

public bool IsReadOnly => false;

#endregion

#region IList<T>

public int IndexOf(T item)
{
return Array.IndexOf(_snapshot, item);
}

public void Insert(int index, T item)
{
lock (_lock)
{
_list.Insert(index, item);
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, index));
}
}

public void RemoveAt(int index)
{
lock (_lock)
{
var item = _list[index];
_list.RemoveAt(index);
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
}
}


public T this[int index]
{
get => _snapshot[index];
set
{
lock (_lock)
{
var item = _list[index];
_list[index] = value;
updateSnapshot();

notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace, value, item, index));
}
}
}
#endregion

#region IList, ICollection (explicit implementations)
//omitted
#endregion
}
}









share|improve this question









New contributor




mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.

























    1














    I'm working on a WPF 4.5 desktop application that has several nested collections with the following key aspects:




    • writes mainly from UI thread, but also from worker threads

    • writes are relatively seldom, mainly directly after a user interaction

    • reads from any thread, but esp. one performance critical worker thread with very many reads (iterating in millisecond intervals)

    • with the exception of the thread mentioned above, read/write performance should not be critical

    • items need to be ordered, i.e. an item's position must always stay the same

    • "remove item" must be supported

    • "insert at" must be supported, although I'm aware that indices must be handled with care, if several threads are involved

    • collection will be used as a WPF binding source and must be observable (implement INotifyCollectionChanged and INotifyPropertyChanged so WPF can update the UI, if items are added/removed)


    • collection must support live-shaping (allowing WPF to instantly update a control's sorting/filtering, if relevant items' properties change; requires the underlying collection to implement IList or similar, so a ListCollectionView can be used)

    • a lookup via key is not required (or can be achieved using extension methods, e.g. FirstOrDefault)

    • approx. max. number of collections < 10k

    • approx. max. number of items / collection < 1k


    The out-of-the-box system classes have the following issues (for my use-case) which prevent me from using them as-is:





    • System.Collections.Concurrent classes do not implement IList - and cannot be used for live-shaping


    • System.Collections.ObjectModel.ObservableCollection<T> is not thread-safe


    So to fulfill all above requirements I created a wrapper class that implements the required interfaces (e.g. IList, INotifyCollectionChanged...). Internally I chose to use List<T>. (I could have chosen ObservableCollection, but I wanted full control when invoking/dispatching CollectionChanged.)



    For all write operations the wrapper class uses lock(_lock) and delegates the call to the inner list. Also - from within the lock - it updates an Array snapshot of the current list, stored in a private field, _snapshot. Then - still from within the lock - it uses System.Windows.Threading.Dispatcher.InvokeAsync() to raise the CollectionChanged event on the correct UI thread.



    All read operations use the cached _snapshot, esp. GetEnumerator. The intention behind the snapshot is to avoid locking in the GetEnumerator implementation, for performance reasons of the thread with many reads.



    Is the approach ok, what am I missing, what else must I be aware of?



    Here's my current code (with some omissions), which appears to work:



    using System;
    using System.Collections;
    using System.Collections.Generic;
    using System.Collections.Specialized;
    using System.ComponentModel;
    using System.Linq;
    using System.Threading;
    using System.Windows;

    namespace StackOverflow.Questions
    {
    public class ObservableConcurrentList<T> : IList, IList<T>, INotifyCollectionChanged, INotifyPropertyChanged
    {
    private readonly System.Windows.Threading.Dispatcher _context;
    private readonly IList<T> _list = new List<T>();
    private readonly object _lock = new object();
    private T _snapshot;

    public ObservableConcurrentList()
    {
    _context = Application.Current?.Dispatcher;

    updateSnapshot();

    SuppressNotifications = suppressNotifications;
    }

    public event NotifyCollectionChangedEventHandler CollectionChanged;

    public event PropertyChangedEventHandler PropertyChanged;

    private void updateSnapshot()
    {
    lock (_lock) //precautionary; should be re-entry
    {
    Interlocked.Exchange(ref _snapshot, _list.ToArray());
    }
    }

    private void notify(NotifyCollectionChangedEventArgs args)
    {
    if (_context == null)
    {
    invokeCollectionChanged(args);
    }
    else
    {
    _context.InvokeAsync(() => invokeCollectionChanged(args));
    }
    }

    private void invokeCollectionChanged(NotifyCollectionChangedEventArgs args)
    {
    CollectionChanged?.Invoke(this, args);
    PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Count)));
    }

    #region IEnumerable
    public IEnumerator<T> GetEnumerator()
    {
    var localSnapshot = _snapshot; //create local variable to protect enumerator, if class member (_snapshot) should be changed/replaced while iterating
    return ((IEnumerable<T>)localSnapshot).GetEnumerator();
    }

    IEnumerator IEnumerable.GetEnumerator()
    {
    return GetEnumerator();
    }
    #endregion

    #region ICollection<T>
    public void Add(T item)
    {
    lock (_lock)
    {
    _list.Add(item);
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, _list.Count - 1));
    }
    }

    public bool Contains(T item)
    {
    return _snapshot.Contains(item);
    }

    public void CopyTo(T array, int arrayIndex)
    {
    _snapshot.CopyTo(array, arrayIndex);
    }

    public bool Remove(T item)
    {
    lock (_lock)
    {
    var index = _list.IndexOf(item);
    if (index > -1)
    {
    if (_list.Remove(item))
    {
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
    return true;
    }
    }

    return false;
    }
    }

    public void Clear()
    {
    lock (_lock)
    {
    _list.Clear();
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
    }
    }

    public bool IsReadOnly => false;

    #endregion

    #region IList<T>

    public int IndexOf(T item)
    {
    return Array.IndexOf(_snapshot, item);
    }

    public void Insert(int index, T item)
    {
    lock (_lock)
    {
    _list.Insert(index, item);
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, index));
    }
    }

    public void RemoveAt(int index)
    {
    lock (_lock)
    {
    var item = _list[index];
    _list.RemoveAt(index);
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
    }
    }


    public T this[int index]
    {
    get => _snapshot[index];
    set
    {
    lock (_lock)
    {
    var item = _list[index];
    _list[index] = value;
    updateSnapshot();

    notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace, value, item, index));
    }
    }
    }
    #endregion

    #region IList, ICollection (explicit implementations)
    //omitted
    #endregion
    }
    }









    share|improve this question









    New contributor




    mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
    Check out our Code of Conduct.























      1












      1








      1







      I'm working on a WPF 4.5 desktop application that has several nested collections with the following key aspects:




      • writes mainly from UI thread, but also from worker threads

      • writes are relatively seldom, mainly directly after a user interaction

      • reads from any thread, but esp. one performance critical worker thread with very many reads (iterating in millisecond intervals)

      • with the exception of the thread mentioned above, read/write performance should not be critical

      • items need to be ordered, i.e. an item's position must always stay the same

      • "remove item" must be supported

      • "insert at" must be supported, although I'm aware that indices must be handled with care, if several threads are involved

      • collection will be used as a WPF binding source and must be observable (implement INotifyCollectionChanged and INotifyPropertyChanged so WPF can update the UI, if items are added/removed)


      • collection must support live-shaping (allowing WPF to instantly update a control's sorting/filtering, if relevant items' properties change; requires the underlying collection to implement IList or similar, so a ListCollectionView can be used)

      • a lookup via key is not required (or can be achieved using extension methods, e.g. FirstOrDefault)

      • approx. max. number of collections < 10k

      • approx. max. number of items / collection < 1k


      The out-of-the-box system classes have the following issues (for my use-case) which prevent me from using them as-is:





      • System.Collections.Concurrent classes do not implement IList - and cannot be used for live-shaping


      • System.Collections.ObjectModel.ObservableCollection<T> is not thread-safe


      So to fulfill all above requirements I created a wrapper class that implements the required interfaces (e.g. IList, INotifyCollectionChanged...). Internally I chose to use List<T>. (I could have chosen ObservableCollection, but I wanted full control when invoking/dispatching CollectionChanged.)



      For all write operations the wrapper class uses lock(_lock) and delegates the call to the inner list. Also - from within the lock - it updates an Array snapshot of the current list, stored in a private field, _snapshot. Then - still from within the lock - it uses System.Windows.Threading.Dispatcher.InvokeAsync() to raise the CollectionChanged event on the correct UI thread.



      All read operations use the cached _snapshot, esp. GetEnumerator. The intention behind the snapshot is to avoid locking in the GetEnumerator implementation, for performance reasons of the thread with many reads.



      Is the approach ok, what am I missing, what else must I be aware of?



      Here's my current code (with some omissions), which appears to work:



      using System;
      using System.Collections;
      using System.Collections.Generic;
      using System.Collections.Specialized;
      using System.ComponentModel;
      using System.Linq;
      using System.Threading;
      using System.Windows;

      namespace StackOverflow.Questions
      {
      public class ObservableConcurrentList<T> : IList, IList<T>, INotifyCollectionChanged, INotifyPropertyChanged
      {
      private readonly System.Windows.Threading.Dispatcher _context;
      private readonly IList<T> _list = new List<T>();
      private readonly object _lock = new object();
      private T _snapshot;

      public ObservableConcurrentList()
      {
      _context = Application.Current?.Dispatcher;

      updateSnapshot();

      SuppressNotifications = suppressNotifications;
      }

      public event NotifyCollectionChangedEventHandler CollectionChanged;

      public event PropertyChangedEventHandler PropertyChanged;

      private void updateSnapshot()
      {
      lock (_lock) //precautionary; should be re-entry
      {
      Interlocked.Exchange(ref _snapshot, _list.ToArray());
      }
      }

      private void notify(NotifyCollectionChangedEventArgs args)
      {
      if (_context == null)
      {
      invokeCollectionChanged(args);
      }
      else
      {
      _context.InvokeAsync(() => invokeCollectionChanged(args));
      }
      }

      private void invokeCollectionChanged(NotifyCollectionChangedEventArgs args)
      {
      CollectionChanged?.Invoke(this, args);
      PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Count)));
      }

      #region IEnumerable
      public IEnumerator<T> GetEnumerator()
      {
      var localSnapshot = _snapshot; //create local variable to protect enumerator, if class member (_snapshot) should be changed/replaced while iterating
      return ((IEnumerable<T>)localSnapshot).GetEnumerator();
      }

      IEnumerator IEnumerable.GetEnumerator()
      {
      return GetEnumerator();
      }
      #endregion

      #region ICollection<T>
      public void Add(T item)
      {
      lock (_lock)
      {
      _list.Add(item);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, _list.Count - 1));
      }
      }

      public bool Contains(T item)
      {
      return _snapshot.Contains(item);
      }

      public void CopyTo(T array, int arrayIndex)
      {
      _snapshot.CopyTo(array, arrayIndex);
      }

      public bool Remove(T item)
      {
      lock (_lock)
      {
      var index = _list.IndexOf(item);
      if (index > -1)
      {
      if (_list.Remove(item))
      {
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
      return true;
      }
      }

      return false;
      }
      }

      public void Clear()
      {
      lock (_lock)
      {
      _list.Clear();
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
      }
      }

      public bool IsReadOnly => false;

      #endregion

      #region IList<T>

      public int IndexOf(T item)
      {
      return Array.IndexOf(_snapshot, item);
      }

      public void Insert(int index, T item)
      {
      lock (_lock)
      {
      _list.Insert(index, item);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, index));
      }
      }

      public void RemoveAt(int index)
      {
      lock (_lock)
      {
      var item = _list[index];
      _list.RemoveAt(index);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
      }
      }


      public T this[int index]
      {
      get => _snapshot[index];
      set
      {
      lock (_lock)
      {
      var item = _list[index];
      _list[index] = value;
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace, value, item, index));
      }
      }
      }
      #endregion

      #region IList, ICollection (explicit implementations)
      //omitted
      #endregion
      }
      }









      share|improve this question









      New contributor




      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.











      I'm working on a WPF 4.5 desktop application that has several nested collections with the following key aspects:




      • writes mainly from UI thread, but also from worker threads

      • writes are relatively seldom, mainly directly after a user interaction

      • reads from any thread, but esp. one performance critical worker thread with very many reads (iterating in millisecond intervals)

      • with the exception of the thread mentioned above, read/write performance should not be critical

      • items need to be ordered, i.e. an item's position must always stay the same

      • "remove item" must be supported

      • "insert at" must be supported, although I'm aware that indices must be handled with care, if several threads are involved

      • collection will be used as a WPF binding source and must be observable (implement INotifyCollectionChanged and INotifyPropertyChanged so WPF can update the UI, if items are added/removed)


      • collection must support live-shaping (allowing WPF to instantly update a control's sorting/filtering, if relevant items' properties change; requires the underlying collection to implement IList or similar, so a ListCollectionView can be used)

      • a lookup via key is not required (or can be achieved using extension methods, e.g. FirstOrDefault)

      • approx. max. number of collections < 10k

      • approx. max. number of items / collection < 1k


      The out-of-the-box system classes have the following issues (for my use-case) which prevent me from using them as-is:





      • System.Collections.Concurrent classes do not implement IList - and cannot be used for live-shaping


      • System.Collections.ObjectModel.ObservableCollection<T> is not thread-safe


      So to fulfill all above requirements I created a wrapper class that implements the required interfaces (e.g. IList, INotifyCollectionChanged...). Internally I chose to use List<T>. (I could have chosen ObservableCollection, but I wanted full control when invoking/dispatching CollectionChanged.)



      For all write operations the wrapper class uses lock(_lock) and delegates the call to the inner list. Also - from within the lock - it updates an Array snapshot of the current list, stored in a private field, _snapshot. Then - still from within the lock - it uses System.Windows.Threading.Dispatcher.InvokeAsync() to raise the CollectionChanged event on the correct UI thread.



      All read operations use the cached _snapshot, esp. GetEnumerator. The intention behind the snapshot is to avoid locking in the GetEnumerator implementation, for performance reasons of the thread with many reads.



      Is the approach ok, what am I missing, what else must I be aware of?



      Here's my current code (with some omissions), which appears to work:



      using System;
      using System.Collections;
      using System.Collections.Generic;
      using System.Collections.Specialized;
      using System.ComponentModel;
      using System.Linq;
      using System.Threading;
      using System.Windows;

      namespace StackOverflow.Questions
      {
      public class ObservableConcurrentList<T> : IList, IList<T>, INotifyCollectionChanged, INotifyPropertyChanged
      {
      private readonly System.Windows.Threading.Dispatcher _context;
      private readonly IList<T> _list = new List<T>();
      private readonly object _lock = new object();
      private T _snapshot;

      public ObservableConcurrentList()
      {
      _context = Application.Current?.Dispatcher;

      updateSnapshot();

      SuppressNotifications = suppressNotifications;
      }

      public event NotifyCollectionChangedEventHandler CollectionChanged;

      public event PropertyChangedEventHandler PropertyChanged;

      private void updateSnapshot()
      {
      lock (_lock) //precautionary; should be re-entry
      {
      Interlocked.Exchange(ref _snapshot, _list.ToArray());
      }
      }

      private void notify(NotifyCollectionChangedEventArgs args)
      {
      if (_context == null)
      {
      invokeCollectionChanged(args);
      }
      else
      {
      _context.InvokeAsync(() => invokeCollectionChanged(args));
      }
      }

      private void invokeCollectionChanged(NotifyCollectionChangedEventArgs args)
      {
      CollectionChanged?.Invoke(this, args);
      PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Count)));
      }

      #region IEnumerable
      public IEnumerator<T> GetEnumerator()
      {
      var localSnapshot = _snapshot; //create local variable to protect enumerator, if class member (_snapshot) should be changed/replaced while iterating
      return ((IEnumerable<T>)localSnapshot).GetEnumerator();
      }

      IEnumerator IEnumerable.GetEnumerator()
      {
      return GetEnumerator();
      }
      #endregion

      #region ICollection<T>
      public void Add(T item)
      {
      lock (_lock)
      {
      _list.Add(item);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, _list.Count - 1));
      }
      }

      public bool Contains(T item)
      {
      return _snapshot.Contains(item);
      }

      public void CopyTo(T array, int arrayIndex)
      {
      _snapshot.CopyTo(array, arrayIndex);
      }

      public bool Remove(T item)
      {
      lock (_lock)
      {
      var index = _list.IndexOf(item);
      if (index > -1)
      {
      if (_list.Remove(item))
      {
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
      return true;
      }
      }

      return false;
      }
      }

      public void Clear()
      {
      lock (_lock)
      {
      _list.Clear();
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
      }
      }

      public bool IsReadOnly => false;

      #endregion

      #region IList<T>

      public int IndexOf(T item)
      {
      return Array.IndexOf(_snapshot, item);
      }

      public void Insert(int index, T item)
      {
      lock (_lock)
      {
      _list.Insert(index, item);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, index));
      }
      }

      public void RemoveAt(int index)
      {
      lock (_lock)
      {
      var item = _list[index];
      _list.RemoveAt(index);
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove, item, index));
      }
      }


      public T this[int index]
      {
      get => _snapshot[index];
      set
      {
      lock (_lock)
      {
      var item = _list[index];
      _list[index] = value;
      updateSnapshot();

      notify(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace, value, item, index));
      }
      }
      }
      #endregion

      #region IList, ICollection (explicit implementations)
      //omitted
      #endregion
      }
      }






      c# multithreading concurrency wpf






      share|improve this question









      New contributor




      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.











      share|improve this question









      New contributor




      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.









      share|improve this question




      share|improve this question








      edited 13 mins ago









      200_success

      128k15150412




      128k15150412






      New contributor




      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.









      asked 3 hours ago









      mike

      1062




      1062




      New contributor




      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.





      New contributor





      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.






      mike is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
      Check out our Code of Conduct.



























          active

          oldest

          votes











          Your Answer





          StackExchange.ifUsing("editor", function () {
          return StackExchange.using("mathjaxEditing", function () {
          StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
          StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
          });
          });
          }, "mathjax-editing");

          StackExchange.ifUsing("editor", function () {
          StackExchange.using("externalEditor", function () {
          StackExchange.using("snippets", function () {
          StackExchange.snippets.init();
          });
          });
          }, "code-snippets");

          StackExchange.ready(function() {
          var channelOptions = {
          tags: "".split(" "),
          id: "196"
          };
          initTagRenderer("".split(" "), "".split(" "), channelOptions);

          StackExchange.using("externalEditor", function() {
          // Have to fire editor after snippets, if snippets enabled
          if (StackExchange.settings.snippets.snippetsEnabled) {
          StackExchange.using("snippets", function() {
          createEditor();
          });
          }
          else {
          createEditor();
          }
          });

          function createEditor() {
          StackExchange.prepareEditor({
          heartbeatType: 'answer',
          autoActivateHeartbeat: false,
          convertImagesToLinks: false,
          noModals: true,
          showLowRepImageUploadWarning: true,
          reputationToPostImages: null,
          bindNavPrevention: true,
          postfix: "",
          imageUploader: {
          brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
          contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
          allowUrls: true
          },
          onDemand: true,
          discardSelector: ".discard-answer"
          ,immediatelyShowMarkdownHelp:true
          });


          }
          });






          mike is a new contributor. Be nice, and check out our Code of Conduct.










          draft saved

          draft discarded


















          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210575%2fconcurrent-observable-collection%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown






























          active

          oldest

          votes













          active

          oldest

          votes









          active

          oldest

          votes






          active

          oldest

          votes








          mike is a new contributor. Be nice, and check out our Code of Conduct.










          draft saved

          draft discarded


















          mike is a new contributor. Be nice, and check out our Code of Conduct.













          mike is a new contributor. Be nice, and check out our Code of Conduct.












          mike is a new contributor. Be nice, and check out our Code of Conduct.
















          Thanks for contributing an answer to Code Review Stack Exchange!


          • Please be sure to answer the question. Provide details and share your research!

          But avoid



          • Asking for help, clarification, or responding to other answers.

          • Making statements based on opinion; back them up with references or personal experience.


          Use MathJax to format equations. MathJax reference.


          To learn more, see our tips on writing great answers.





          Some of your past answers have not been well-received, and you're in danger of being blocked from answering.


          Please pay close attention to the following guidance:


          • Please be sure to answer the question. Provide details and share your research!

          But avoid



          • Asking for help, clarification, or responding to other answers.

          • Making statements based on opinion; back them up with references or personal experience.


          To learn more, see our tips on writing great answers.




          draft saved


          draft discarded














          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210575%2fconcurrent-observable-collection%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown





















































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown

































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown







          Popular posts from this blog

          Morgemoulin

          Scott Moir

          Souastre