Class SafeDictionary<TKey, TValue>
Represents a thread-safe collection of key/value pairs that can be modified by multiple threads concurrently.
public class SafeDictionary<TKey, TValue> : Hashtable, ICloneThreadItem, IGeneralIndexable<TKey, TValue>, IGeneralIndexable
Type Parameters
TKey
Key type
TValue
Value type
- Inheritance
-
objectHashtableSafeDictionary<TKey, TValue>
- Implements
-
IGeneralIndexable<TKey, TValue>
- Derived
- Extension Methods
Remarks
Keep in mind that the operations of this class are thread-safe. However, a sequence of operations inherently cannot be thread-safe. If you want to perform multiple operations as one atomic operation, use the SyncRoot object.
In current implementation some read operations do not require locking (i.e. multiple threads can read even though some other thread has the lock). If your situation requires all the readers to wait when the dictionary is locked by some other thread, consider some other collection suitable for readers-writers synchronization problem.
Constructors
SafeDictionary()
SafeDictionary constructor.
public SafeDictionary()
SafeDictionary(IDictionary, IEqualityComparer)
Constructor.
public SafeDictionary(IDictionary d, IEqualityComparer comparer = null)
Parameters
d
IDictionarySource data dictionary
comparer
IEqualityComparerEquality comparer for the items
SafeDictionary(IEqualityComparer)
SafeDictionary constructor.
public SafeDictionary(IEqualityComparer comparer)
Parameters
comparer
IEqualityComparerEquality comparer for the items
Fields
mDefaultValue
Default value.
protected TValue mDefaultValue
Field Value
- TValue
mNullValue
Null value.
protected object mNullValue
Field Value
- object
mUseWeakReferences
If true, the weak references are used for the items so the memory can be cleared upon request
protected bool mUseWeakReferences
Field Value
- bool
Properties
AllowNulls
If true, the dictionary allows null values as valid.
public bool AllowNulls { get; set; }
Property Value
- bool
CopyToNewThread
If true, the dictionary is copied (not cloned) to a new thread
public bool CopyToNewThread { get; set; }
Property Value
- bool
DefaultValue
Default value.
public TValue DefaultValue { get; set; }
Property Value
- TValue
IsSynchronized
Returns true if dictionary is synchronized.
public override bool IsSynchronized { get; }
Property Value
- bool
Remarks
Current implementation ensures thread-safety for read/write operations (read operations are safe to be performed without the need for locking). To synchronize enumeration or multiple operations use SyncRoot.
this[object]
Items indexer. Gets or sets the value in the dictionary.
public override object this[object key] { get; set; }
Parameters
key
objectValue key
Property Value
- object
Remarks
The this[object].get operation does not require locking for concurrent access.
this[TKey]
Items indexer. Gets or sets the value in the dictionary.
public virtual TValue this[TKey key] { get; set; }
Parameters
key
TKeyValue key
Property Value
- TValue
Remarks
The this[TKey].get operation does not require locking for concurrent access.
Keys
Gets an Collection containing the keys in the System.Collections.Hashtable.
public override ICollection Keys { get; }
Property Value
- ICollection
NullValue
Null value.
public object NullValue { get; protected set; }
Property Value
- object
SyncRoot
Gets an object that can be used to synchronize access to the SafeDictionary<TKey, TValue>.
public override object SyncRoot { get; }
Property Value
- object
An object that can be used to synchronize access to the SafeDictionary<TKey, TValue>.
Examples
Use the following snippet to enumerate the dictionary entries in a thread-safe way.
lock (dictionary.SyncRoot)
{
foreach (var entry in dictionary)
{
// Perform some operation
// Try to avoid time consuming operations to release the lock as soon as possible
// No other thread can modify the dictionary when it is locked using the SyncRoot property
// However, read access does not require locking, therefore any other thread can still read the dictionary's content
}
}
Remarks
Utilize this property when you need to perform multiple dictionary operations as a single atomic operation, or when you need to enumerate the entries.
Obtaining a lock does not prevent all other threads from read operations, since most read operations do not need to use locking.
TypedKeys
Gets a typed collection of keys in this dictionary
public IEnumerable<TKey> TypedKeys { get; }
Property Value
- IEnumerable<TKey>
TypedValues
Gets a typed collection of values in this dictionary
public IEnumerable<TValue> TypedValues { get; }
Property Value
- IEnumerable<TValue>
UseWeakReferences
If true, the weak references are used for the items so the memory can be cleared upon request. The property can be set only when the dictionary is empty. To ensure thread-safety, you have to perform the check for emptiness and property assignment in a critical section (use SyncRoot for that purpose).
public bool UseWeakReferences { get; set; }
Property Value
- bool
Examples
Use the following code snippet to ensure thread-safety while changing this property.
lock (dictionary.SyncRoot)
{
dictionary.Clear();
dictionary.UseWeakReferences = true;
}
Values
Gets a Collection containing the values in the System.Collections.Hashtable.
public override ICollection Values { get; }
Property Value
- ICollection
Methods
Add(object, object)
Adds the value to the dictionary if it does not exist. Updates existing, if it does exist.
public override void Add(object key, object value)
Parameters
key
objectKey
value
objectValue
Examples
If you want to add some key-value pair only if it does not exist within the dictionary, use the following snippet to do so in a thread-safe way.
if (!dictionary.ContainsKey(myKey))
{
lock (dictionary.SyncRoot)
{
if (!dictionary.ContainsKey(myKey))
{
dictionary.Add(myKey, myValue);
}
}
}
AddMultiple(string[], bool)
Adds multiple items with same value to the dictionary
public void AddMultiple(string[] items, bool value)
Parameters
items
string[]Items to add
value
boolItems value
Examples
Use the following snippet to make sure no other thread modifies the dictionary while
it is being populated by items
.
string[] items = GetItems(...);
lock (dictionary.SyncRoot)
{
dictionary.AddMultiple(items, true);
// All the items are set to true now
}
The following approach is also thread-safe, but the result may differ from the previous one if another thread performs any write operation while the items are being added.
string[] items = GetItems(...);
dictionary.AddMultiple(items, true);
// Any other thread might have modified some entry (i.e. the items are not guaranteed to be set to true)
Remarks
The operation is not performed in a critical section. If you want to make sure no other thread modifies the dictionary (e.g. adds/removes an item) while the dictionary is being filled with items, use the SyncRoot.
Clear()
Removes all elements from the System.Collections.Hashtable.
public override void Clear()
Clone()
Clones the dictionary
public override object Clone()
Returns
- object
CloneForNewThread()
Clones the object for new thread
public object CloneForNewThread()
Returns
- object
Contains(object)
Returns true if the dictionary contains the given key
public override bool Contains(object key)
Parameters
key
objectKey
Returns
- bool
Remarks
This operation does not require locking for concurrent access.
ContainsKey(object)
Returns true if the dictionary contains the given key
public override bool ContainsKey(object key)
Parameters
key
objectKey
Returns
- bool
Remarks
This operation does not require locking for concurrent access.
ContainsValue(object)
Returns true if the dictionary contains the given value
public override bool ContainsValue(object value)
Parameters
value
objectValue
Returns
- bool
CopyPropertiesTo(SafeDictionary<TKey, TValue>)
Copies the dictionary properties to the target dictionary
protected void CopyPropertiesTo(SafeDictionary<TKey, TValue> target)
Parameters
target
SafeDictionary<TKey, TValue>Target dictionary
CopyTo(Array, int)
Copies the System.Collections.Hashtable elements to a one-dimensional System.Array instance at the specified index.
public override void CopyTo(Array array, int arrayIndex)
Parameters
array
ArrayThe one-dimensional System.Array that is the destination of the System.Collections.DictionaryEntry objects copied from System.Collections.Hashtable. The System.Array must have zero-based indexing.
arrayIndex
intThe zero-based index in array at which copying begins.
Examples
Use the following code snippet to ensure the proper array size.
lock (dictionary.SyncRoot)
{
// The lock makes sure the entries count does not change between the array allocation and the copy operation
// The GetArrayOfSize is only an illustrational method
Array destinationArray = GetArrayOfSize(dictionary.Count);
dictionary.CopyTo(destinationArray, 0);
}
Remarks
The array
has to be large enough to accommodate all the dictionary entries. Unless the dictionary has a fixed number of entries
(or the entries count is limited somehow and the limit is a known number), you should perform the array allocation in a thread-safe way.
GetInternalValue(TKey)
Gets the value from the internal dictionary
protected object GetInternalValue(TKey key)
Parameters
key
TKeyObject key
Returns
- object
GetRealCount()
Gets the real count of the objects in the dictionary.
public int GetRealCount()
Returns
- int
Examples
lock (dictionary.SyncRoot)
{
int realCount = dictionary.GetRealCount();
// The realCount corresponds to current dictionary's state only when weak references are NOT used
// Otherwise, some entries might have been released from the memory regardless of the lock
}
Remarks
The dictionary's content can be changed by other threads after computing the result. If you want to make sure the dictionary's content does not change until the real count is used, use the SyncRoot.
Important: When the dictionary uses weak references, no lock can guarantee you that no item has been released from the memory while computing the real count.
Remove(object)
Removes the element with the specified key from the System.Collections.Hashtable.
public override void Remove(object key)
Parameters
key
objectThe key of the element to remove.
SetInternalValue(TKey, object)
Sets the value in the internal dictionary
protected void SetInternalValue(TKey key, object value)
Parameters
key
TKeyObject key
value
objectObject value
TryGetValue(TKey, out TValue)
Tries to get the value, returns true if the retrieval was successful.
public bool TryGetValue(TKey key, out TValue value)
Parameters
key
TKeyValue key
value
TValueReturning value
Returns
- bool
Remarks
This operation does not require locking for concurrent access.