I'm using a dictionary inside of some Task.
Logically I have set it up so that my Keys will never clash, though sometimes when I am adding to the dictionary I get this Exception.
Index was outside the bounds of the array.
at System.Collections.Generic.Dictionary`2.Insert(TKey key, TValue value, Boolean add)
at System.Collections.Generic.Dictionary`2.Add(TKey key, TValue value)
at Rpc.<MapIntoRpc>b__4[T](Object x) in Rpc.cs:line 113
at System.Threading.Tasks.Task`1.InvokeFuture(Object futureAsObj)
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
I understand there can be concurrency issues from trying to remove or add the same Key multiple times, but I have accounted for that algorithmically.
What causes the add to sometimes fail? What is the best way to work around that?
You should have looked to the documentation. That what it says:
A Dictionary can support multiple readers concurrently, as long as the collection is not modified. Even so, enumerating through a collection is intrinsically not a thread-safe procedure. In the rare case where an enumeration contends with write accesses, the collection must be locked during the entire enumeration. To allow the collection to be accessed by multiple threads for reading and writing, you must implement your own synchronization. For a thread-safe alternative, see ConcurrentDictionary.
Your issue is most likely synchronization. When a Dictionary is added to it sometimes needs to increase the size of the underlying structure (an array). If you are adding from multiple threads that may result in an IndexOutOfRangeException
. You need to use locks etc. to make sure you are adding in a safe way.
Alternatively you can use a ConcurrentDictionary which is a thread-safe collection.
So you might think Whatever! it will just break the one time
- but nope:
There goes three hours of sales (until IIS recycled on a schedule) because of a dictionary added for debugging purposes that wasn't ever even being read from.
Note: This was running for 3.5 years before I hit this condition.
private Dictionary<string, string> _debugLookup;
_debugLookup[key] = virtualPath;
This wasn't even a static dictionary - it was an MVC IViewLocationCache
that was an instance method.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With