简化ReaderWriterLockSlim语法

时间:2011-07-20 08:08:49

标签: .net multithreading syntax simplify readerwriterlockslim

我最近不得不使用ReaderWriterLockSlim来同步对多个线程之间共享的多个资源的访问。在这样做时,我觉得使用ReaderWriterLockSlim并不是特别容易,特别是当你必须在多个地方使用它时。您必须拥有try...finally块,并记住打开和关闭锁。在许多情况下,我还发现自己打开写锁并关闭读锁而不是关闭写锁。因此,我试图想出一种更简单的方法来使用ReaderWriterLockSlim。这是我得到的地方

class Locked<T>
{
    private T _resource;
    private ReaderWriterLockSlim _lock;

    public Locked(T resource)
    {
        _resource = resource;
        _lock = new ReaderWriterLockSlim();
    }

    public void Read(Action<T> ReadAction)
    {
        try
        {
            _lock.EnterReadLock();
            ReadAction(_resource);
        }
        finally
        {
            _lock.ExitReadLock();
        }
    }

    public void Write(Action<T> WriteAction)
    {
        try
        {
            _lock.EnterWriteLock();
            WriteAction(_resource);
        }
        finally
        {
            _lock.ExitWriteLock();
        }
    }
}

现在,例如如果我们需要同步访问List<string>这就是我们使用上面的类

的方法
public class Demo
{
    private Locked<List<string>> _listOfString;

    public Demo()
    {
        _listOfString = new Locked<List<string>>(new List<string>());
    }

    public void writeMethod(string value)
    {
        _listOfString.Write(list =>
        {
            list.Add(value);
        });
    }

    public string readMethod(int index)
    {
        string value = null;
        _listOfString.Read(list =>
        {
            value = list[index];
        });
        return value;
    }
}

你认为这种方法更好吗?是否有任何缺点或缺陷。

1 个答案:

答案 0 :(得分:4)

我宁愿将锁定逻辑包装在IDisposable中,并将要锁定的代码放在using块中,与lock语句非常相似:

class DisposableLock : IDisposable
{
    Action _exitLock;
    public DisposableLock(Action exitLock)
    {
        _exitLock = exitLock;
    }

    public void Dispose()
    {
        _exitLock();
    }
}

class ReadLock : DisposableLock
{
    public ReadLock(ReaderWriterLockSlim slimLock)
        : base(() => slimLock.ExitReadLock())
    {
        slimLock.EnterReadLock();
    }
}

class WriteLock : DisposableLock
{
    public WriteLock(ReaderWriterLockSlim slimLock)
        : base(() => slimLock.ExitWriteLock())
    {
        slimLock.EnterWriteLock();
    }
}

您可以这样使用它:

using(new ReadLock(_lock))
{
    // ... your synchronized operation
}

using(new WriteLock(_lock))
{
    // ... your synchronized operation
}