Table of Contents

Class SafeDictionary<TKey, TValue>

Namespace
CMS.Base
Assembly
CMS.Base.dll

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
object
Hashtable
SafeDictionary<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 IDictionary

Source data dictionary

comparer IEqualityComparer

Equality comparer for the items

SafeDictionary(IEqualityComparer)

SafeDictionary constructor.

public SafeDictionary(IEqualityComparer comparer)

Parameters

comparer IEqualityComparer

Equality 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 object

Value 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 TKey

Value 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 object

Key

value object

Value

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 bool

Items 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 object

Key

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 object

Key

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 object

Value

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 Array

The 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 int

The 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 TKey

Object 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 object

The 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 TKey

Object key

value object

Object 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 TKey

Value key

value TValue

Returning value

Returns

bool

Remarks

This operation does not require locking for concurrent access.