你的实现是正确的。不幸的是,.NET Framework 不提供内置的并发哈希集类型。但是,有一些解决方法。
并发字典(推荐)
这第一个是使用ConcurrentDictionary<TKey, TValue>
命名空间中的类System.Collections.Concurrent
。在这种情况下,该值是没有意义的,所以我们可以使用一个简单的byte
(内存中的 1 个字节)。
private ConcurrentDictionary<string, byte> _data;
这是推荐的选项,因为该类型是线程安全的,并且为您提供与HashSet<T>
除了键和值是不同对象之外的相同的优点。
资料来源:社交 MSDN
并发包
如果您不介意重复的条目,您可以ConcurrentBag<T>
在与前一个类相同的命名空间中使用该类。
private ConcurrentBag<string> _data;
自我实现
最后,正如您所做的那样,您可以使用锁或 .NET 为您提供的其他线程安全方式来实现自己的数据类型。这是一个很好的例子:How to implement ConcurrentHashSet in .Net
此解决方案的唯一缺点是该类型HashSet<T>
没有正式的并发访问,即使对于读取操作也是如此。
我引用了链接帖子的代码(最初由Ben Mosher编写)。
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
}
}
编辑:将入口锁定方法移到try
块之外,因为它们可能引发异常并执行finally
块中包含的指令。