3

我有一些使用 ReaderWriterLockSlim 的代码。当构建某个对象时,我会在其上获取写锁,并在稍后处理该对象时释放它。但是,由于这些调用的来源,我不能保证它们会在同一个线程上,这是 ReaderWriterLockSlim 的要求。

我相信一个合理的解决方案是在专用线程上运行对象的构造和处置,并让调用代码等待该任务完成(但保持线程处于活动状态)。看起来很乱,但如果不大规模重组我们的代码,我想不出另一种方法。

是否存在允许我在同一个线程上运行两个任务的现有 TaskScheduler 子类?

我当然愿意接受另一种这样做的范式。

4

1 回答 1

0

我有类似的问题,所以我希望我的解决方案也能帮助你。

基本上,问题在于 ReaderWriterLockSlim 具有线程亲和性,这意味着获得锁的线程是唯一可以释放它的线程。

解决方案是再创建一个专用线程,但与您的建议相反,该线程将专用于获取和释放锁。

我猜你的代码看起来像这样:

public class ClassUsingReaderWriterLockSlim
{
    private ReaderWriterLockSlim rwsLock;

    public void MethodThatAcquiresLock()
    {
        rwsLock.EnterWriteLock();
    }

    public void MethodThatReleasesLock()
    {
        rwsLock.ExitWriteLock();
    }
}

解决您的问题的代码如下所示:

public class ClassUsingReaderWriterLockSlim
{
    private ReaderWriterLockSlim rwsLock;
    Thread dedicatedThreadForReaderWriterLockSlim;
    Queue<string> commandsForDedicatedThread;

    public ClassUsingReaderWriterLockSlim()
    {
        commandsForDedicatedThread = new Queue<string>();
        dedicatedThreadForReaderWriterLockSlim = new Thread(ThreadFunction);
        dedicatedThreadForReaderWriterLockSlim.Start();
    }

    private void ThreadFunction(object obj)
    {
        while (!terminatingCondition)
        {
            // Wait until something is in queue...

            if (commandsForDedicatedThread.Count > 0)
            {
                switch (commandsForDedicatedThread.Dequeue())
                {
                    case "ENTER LOCK":
                        rwsLock.EnterWriteLock();
                    case "EXIT LOCK":
                        rwsLock.EnterWriteLock();
                    default:
                    // Do nothing...
                }
            }
        }
    }

    public void MethodThatAcquiresLock()
    {
        commandsForDedicatedThread.Enqueue("ENTER LOCK");
    }

    public void MethodThatReleasesLock()
    {
        commandsForDedicatedThread.Enqueue("EXIT LOCK");
    }
}

好吧,对于您的生产代码,您可能会有所不同,但基本思想是有专门的线程来进行锁定和解锁,这样,从哪个线程调用到方法并不重要应该锁定和解锁代码/资源...

希望这对您有所帮助。

于 2016-12-07T15:39:12.243 回答