问题 SpinLock抛出SynchronizationLockException


我正在尝试使用SpinLock,但是当我调用SpinLock.Exit()时,即使是单线程控制台应用程序中的这个最基本的代码也会抛出以下异常

System.Threading.SynchronizationLockException was unhandled by user code
  Message=The calling thread does not hold the lock.  Source=mscorlib

这是完整的源代码......

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace ConsoleApplication48
{
    class Program
    {
        static readonly SpinLock SpinLock = new SpinLock();
        static void Main(string[] args)
        {
            bool lockTaken = false;
            try
            {
                SpinLock.Enter(ref lockTaken);
                if (lockTaken)
                    Console.WriteLine("Lock taken");
            }
            finally
            {
                if (lockTaken)
                    SpinLock.Exit();
            }
            Console.WriteLine("Done");
        }
    }
}

8611
2018-06-27 10:27


起源

消息说锁不是由调用Exit的同一个线程所拥有 - 但是从代码中可以清楚地知道。 - Peter Morris


答案:


SpinLock是一个结构体,您正在从只读字段中读取它。 C#规范说,在这种情况下,为了调用可能的变异函数,必须将结构复制到可变局部变量。这发生在封面下。

您对进入和退出的呼叫发生在锁的新副本上。因此,Enter正在解锁锁定。

不要让SpinLock变量只读,因为它正在变异。


16
2018-06-27 11:35



啊,所以使用带有值类型的“readonly”不仅指定该字段不可赋值,而且还暗示该值是不可变的,并且为了确保它在读取时复制该值。谢谢 - Peter Morris