Ihre Implementierung ist korrekt. Das .NET Framework bietet leider keinen integrierten gleichzeitigen Hashset-Typ. Es gibt jedoch einige Problemumgehungen.
ConcurrentDictionary (empfohlen)
Diese erste besteht darin, die Klasse ConcurrentDictionary<TKey, TValue>
im Namespace zu verwenden System.Collections.Concurrent
. In diesem Fall ist der Wert sinnlos, sodass wir ein einfaches byte
(1 Byte im Speicher) verwenden können.
private ConcurrentDictionary<string, byte> _data;
Dies ist die empfohlene Option, da der Typ threadsicher ist und Ihnen dieselben Vorteile bietet, als HashSet<T>
wenn ein Ausnahmeschlüssel und ein Wert unterschiedliche Objekte sind.
Quelle: Soziale MSDN
ConcurrentBag
Wenn Ihnen die doppelten Einträge nichts ausmachen, können Sie die Klasse ConcurrentBag<T>
im selben Namespace wie die vorherige Klasse verwenden.
private ConcurrentBag<string> _data;
Selbstimplementierung
Schließlich können Sie wie Sie Ihren eigenen Datentyp implementieren, indem Sie Sperren oder andere Methoden verwenden, die .NET Ihnen bietet, um threadsicher zu sein. Hier ist ein gutes Beispiel: So implementieren Sie ConcurrentHashSet in .Net
Der einzige Nachteil dieser Lösung besteht darin, dass der Typ HashSet<T>
selbst für Lesevorgänge keinen offiziell gleichzeitigen Zugriff bietet.
Ich zitiere den Code des verlinkten Beitrags (ursprünglich von Ben Mosher geschrieben ).
using System;
using System.Collections.Generic;
using System.Threading;
namespace BlahBlah.Utilities
{
public class ConcurrentHashSet<T> : IDisposable
{
private readonly ReaderWriterLockSlim _lock = new ReaderWriterLockSlim(LockRecursionPolicy.SupportsRecursion);
private readonly HashSet<T> _hashSet = new HashSet<T>();
#region Implementation of ICollection<T> ...ish
public bool Add(T item)
{
_lock.EnterWriteLock();
try
{
return _hashSet.Add(item);
}
finally
{
if (_lock.IsWriteLockHeld) _lock.ExitWriteLock();
}
}
public void Clear()
{
_lock.EnterWriteLock();
try
{
_hashSet.Clear();
}
finally
{
if (_lock.IsWriteLockHeld) _lock.ExitWriteLock();
}
}
public bool Contains(T item)
{
_lock.EnterReadLock();
try
{
return _hashSet.Contains(item);
}
finally
{
if (_lock.IsReadLockHeld) _lock.ExitReadLock();
}
}
public bool Remove(T item)
{
_lock.EnterWriteLock();
try
{
return _hashSet.Remove(item);
}
finally
{
if (_lock.IsWriteLockHeld) _lock.ExitWriteLock();
}
}
public int Count
{
get
{
_lock.EnterReadLock();
try
{
return _hashSet.Count;
}
finally
{
if (_lock.IsReadLockHeld) _lock.ExitReadLock();
}
}
}
#endregion
#region Dispose
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
if (disposing)
if (_lock != null)
_lock.Dispose();
}
~ConcurrentHashSet()
{
Dispose(false);
}
#endregion
}
}
BEARBEITEN: Verschieben Sie die Eingangsverriegelungsmethoden außerhalb der try
Blöcke, da sie eine Ausnahme auslösen und die in den finally
Blöcken enthaltenen Anweisungen ausführen können .
System.Collections.Concurrent