I've been using ReaderWriterLockSlim to synchronize reads/writes and it has worked quite well so far.
I have a collection of objects and read/write locks are maintained in a Dictionary<string, ReaderWriterLockSlim>
. I have a scenario where I need to acquire multiple read locks atomically. I'm certain I don't have any code that would allow one thread to attempt to hold more than one write lock concurrently, which makes me think that the code below will work without problems.
If I were trying the approach below with write locks instead of read locks, I'm almost certain I would end up with deadlocks unless I can ensure that locking always happens in the same order (which I can't in my case).
Does anyone see any problems with this code assuming the following:
Any thoughts?
public void WithReaderLocksForItemsNamed(string[] itemNames, Action action)
{
// this gets locks for the items specified from my internation dictionary
ReaderWriterLockSlim[ ] locks = LocksForItemsNamed( itemNames ) ;
try
{
foreach( var @lock in locks )
{
@lock.EnterReadLock( ) ;
}
action( ) ;
}
finally
{
foreach( var @lock in locks )
{
if (@lock.IsReadLockHeld)
{
@lock.ExitReadLock( ) ;
}
}
}
}
This is an old question but I thought it would still be good to update it with the final solution. Although I never experienced problems with the above, I applied the prudent practice of exiting locks in the reverse order they were acquired. Thus the final code looks something like this:
public void WithReaderLocksForItemsNamed(string[] itemNames, Action action)
{
// this gets locks for the items specified from my internation dictionary
ReaderWriterLockSlim[ ] locks = LocksForItemsNamed( itemNames ) ;
try
{
foreach( var @lock in locks )
{
@lock.EnterReadLock( ) ;
}
action( ) ;
}
finally
{
foreach( var @lock in locks.Reverse() )
{
if (@lock.IsReadLockHeld)
{
@lock.ExitReadLock( ) ;
}
}
}
}