การใช้งานของคุณถูกต้อง .NET Framework ไม่ได้ให้ประเภทแฮชเซ็ตที่เกิดขึ้นพร้อมกันในตัวขออภัย อย่างไรก็ตามมีวิธีแก้ไขปัญหาบางอย่าง
ConcurrentDictionary (แนะนำ)
นี้คนแรกคือการใช้ในชั้นเรียนConcurrentDictionary<TKey, TValue>
ในการ System.Collections.Concurrent
namespace ในกรณีนี้ค่าไม่มีจุดหมายดังนั้นเราสามารถใช้วิbyte
(หน่วยความจำ 1 ไบต์)
private ConcurrentDictionary<string, byte> _data;
นี่คือตัวเลือกที่แนะนำเนื่องจากประเภทนี้ปลอดภัยสำหรับเธรดและให้ประโยชน์เหมือนกันHashSet<T>
ยกเว้นคีย์และค่าเป็นวัตถุอื่น
ที่มา: Social MSDN
ConcurrentBag
ถ้าคุณไม่สนใจรายการที่ซ้ำกันคุณสามารถใช้คลาสConcurrentBag<T>
ในเนมสเปซเดียวกันของคลาสก่อนหน้า
private ConcurrentBag<string> _data;
ดำเนินการเอง
ในที่สุดตามที่คุณทำคุณสามารถใช้ชนิดข้อมูลของคุณเองโดยใช้การล็อกหรือวิธีอื่น ๆ ที่. NET ให้คุณปลอดภัยต่อเธรด นี่เป็นตัวอย่างที่ดี: วิธีการใช้ ConcurrentHashSet ใน. 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
บล็อก
System.Collections.Concurrent