c#线程同步使用详解示例
在应用程序中使用多个线程的一个好处是每个线程都可以异步执行。对于 windows 应用程序,耗时的任务可以在后台执行,而使应用程序窗口和控件保持响应。对于服务器应用程序,多线程处理提供了用不同线程处理每个传入请求的能力。否则,在完全满足前一个请求之前,将无法处理每个新请求。然而,线程的异步特性意味着必须协调对资源(如文件句柄、网络连接和内存)的访问。否则,两个或更多的线程可能在同一时间访问相同的资源,而每个线程都不知道其他线程的操作。
线程同步的方式
线程同步有:临界区、互斥区、事件、信号量四种方式
临界区(critical section)、互斥量(mutex)、信号量(semaphore)、事件(event)的区别
1、临界区:通过对多线程的串行化来访问公共资源或一段代码,速度快,适合控制数据访问。在任意时刻只允许一个线程对共享资源进行访问,如果有多个线程试图访问公共资源,那么在有一个线程进入后,其他试图访问公共资源的线程将被挂起,并一直等到进入临界区的线程离开,临界区在被释放后,其他线程才可以抢占。
2、互斥量:采用互斥对象机制。 只有拥有互斥对象的线程才有访问公共资源的权限,因为互斥对象只有一个,所以能保证公共资源不会同时被多个线程访问。互斥不仅能实现同一应用程序的公共资源安全共享,还能实现不同应用程序的公共资源安全共享
3、信号量:它允许多个线程在同一时刻访问同一资源,但是需要限制在同一时刻访问此资源的最大线程数目
4、事 件: 通过通知操作的方式来保持线程的同步,还可以方便实现对多个线程的优先级比较的操作
c#中常见线程同步方法
我们介绍几种常用的c#进行线程同步的方式,这些方式可以根据其原理,找到对应上面的四种类型之一。
1、interlocked
为多个线程共享的变量提供原子操作。
根据经验,那些需要在多线程情况下被保护的资源通常是整型值,且这些整型值在多线程下最常见的操作就是递增、递减或相加操作。interlocked类提供了一个专门的机制用于完成这些特定的操作。这个类提供了increment、decrement、add静态方法用于对int或long型变量的递增、递减或相加操作。此类的方法可以防止可能在下列情况发生的错误:计划程序在某个线程正在更新可由其他线程访问的变量时切换上下文;或者当两个线程在不同的处理器上并发执行时。 此类的成员不引发异常。
increment和decrement方法递增或递减变量并将结果值存储在单个操作中。 在大多数计算机上,增加变量操作不是一个原子操作,需要执行下列步骤:
1)将实例变量中的值加载到寄存器中。
2)增加或减少该值。
3)在实例变量中存储该值。
如果不使用 increment 和 decrement,线程会在执行完前两个步骤后被抢先。 然后由另一个线程执行所有三个步骤。 当第一个线程重新开始执行时,它覆盖实例变量中的值,造成第二个线程执行增减操作的结果丢失。
exchange 方法自动交换指定变量的值。 compareexchange 方法组合了两个操作:比较两个值以及根据比较的结果将第三个值存储在其中一个变量中。 比较和交换操作按原子操作执行。
案例分析:共享打印机。
通常我们会使用共享打印机,几台计算机共享一台打印机,每台计算机可以发出打印指令,可能会出现并发情况。当然我们知道,打印机采用了队列技术。为了简化操作,我们假定,在打印机收到命令时,即可打印,而且在同一时间只能有一个打印任务在执行。我们使用interlocked方法来实现多线程同步。具体代码如下:
using system;
using system.threading;
namespace mutithreadsample.threadsynchronization
{
class printerwithinterlocktest
{
/// <summary>
/// 正在使用的打印机
/// 0代表未使用,1代表正在使用
/// </summary>
public static int usingprinter = 0;
/// <summary>
/// 计算机数量
/// </summary>
public static readonly int computercount = 3;
/// <summary>
/// 测试
/// </summary>
public static void testprint()
{
thread thread;
random random = new random();
for (int i = 0; i < computercount; i++)
{
thread = new thread(mythreadproc);
thread.name = string.format("thread{0}",i);
thread.sleep(random.next(3));
thread.start();
}
}
/// <summary>
/// 线程执行操作
/// </summary>
private static void mythreadproc()
{
//使用打印机进行打印
useprinter();
//当前线程等待1秒
thread.sleep(1000);
}
/// <summary>
/// 使用打印机进行打印
/// </summary>
private static bool useprinter()
{
//检查大引进是否在使用,如果原始值为0,则为未使用,可以进行打印,否则不能打印,继续等待
if (0 == interlocked.exchange(ref usingprinter, 1))
{
console.writeline("{0} acquired the lock", thread.currentthread.name);
//code to access a resource that is not thread safe would go here.
//simulate some work
thread.sleep(500);
console.writeline("{0} exiting lock", thread.currentthread.name);
//释放打印机
interlocked.exchange(ref usingprinter, 0);
return true;
}
else
{
console.writeline(" {0} was denied the lock", thread.currentthread.name);
return false;
}
}
}
}
2、lock 关键字
lock 关键字将语句块标记为临界区,方法是获取给定对象的互斥锁,执行语句,然后释放该锁。
lock 确保当一个线程位于代码的临界区时,另一个线程不进入临界区。如果其他线程试图进入锁定的代码,则它将一直等待(即被阻止),直到该对象被释放。
public void function()
{
system.object locker= new system.object();
lock(locker)
{
// access thread-sensitive resources.
}
}
lock 调用块开始位置的 enter 和块结束位置的 exit。
提供给 lock 关键字的参数必须为基于引用类型的对象,该对象用来定义锁的范围。在上例中,锁的范围限定为此函数,因为函数外不存在任何对该对象的引用。严格地说,提供给 lock 的对象只是用来唯一地标识由多个线程共享的资源,所以它可以是任意类实例。然而,实际上,此对象通常表示需要进行线程同步的资源。例如,如果一个容器对象将被多个线程使用,则可以将该容器传递给 lock,而 lock 后面的同步代码块将访问该容器。只要其他线程在访问该容器前先锁定该容器,则对该对象的访问将是安全同步的。通常,最好避免锁定 public 类型或锁定不受应用程序控制的对象实例,例如,如果该实例可以被公开访问,则 lock(this) 可能会有问题,因为不受控制的代码也可能会锁定该对象。这可能导致死锁,即两个或更多个线程等待释放同一对象。出于同样的原因,锁定公共数据类型(相比于对象)也可能导致问题。锁定字符串尤其危险,因为字符串被公共语言运行库 (clr)“暂留”。这意味着整个程序中任何给定字符串都只有一个实例,就是这同一个对象表示了所有运行的应用程序域的所有线程中的该文本。因此,只要在应用程序进程中的任何位置处具有相同内容的字符串上放置了锁,就将锁定应用程序中该字符串的所有实例。因此,最好锁定不会被暂留的私有或受保护成员。某些类提供专门用于锁定的成员。例如,array 类型提供 syncroot。许多集合类型也提供 syncroot。
常见的结构 lock (this)、lock (typeof (mytype)) 和 lock ("mylock") 违反此准则:
1)如果实例可以被公共访问,将出现 lock (this) 问题。
2)如果 mytype 可以被公共访问,将出现 lock (typeof (mytype)) 问题。
3)由于进程中使用同一字符串的任何其他代码将共享同一个锁,所以出现 lock(“mylock”) 问题。
最佳做法是定义 private 对象来锁定, 或 private static 对象变量来保护所有实例所共有的数据。关于锁的研究,大家可以参考:
案例分析:继续使用共享打印机的案例
我们只需对前面的例子稍作修改即可实现lock进行同步。
声明锁对象:
/// <summary>
/// 正在使用的打印机
/// </summary>
private static object usingprinterlocker = new object();
将打印方法修改如下:
/// <summary>
/// 使用打印机进行打印
/// </summary>
private static void useprinter()
{
//临界区
lock (usingprinterlocker)
{
console.writeline("{0} acquired the lock", thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0} exiting lock", thread.currentthread.name);
}
}
3、监视器
与 lock 关键字类似,监视器防止多个线程同时执行代码块。enter 方法允许一个且仅一个线程继续执行后面的语句;其他所有线程都将被阻止,直到执行语句的线程调用 exit。这与使用 lock 关键字一样。事实上,lock 关键字就是用 monitor 类来实现的。例如:(继续修改共享打印机案例,增加方法useprinterwithmonitor)
/// <summary>
/// 使用打印机进行打印
/// </summary>
private static void useprinterwithmonitor()
{
system.threading.monitor.enter(usingprinterlocker);
try
{
console.writeline("{0} acquired the lock", thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0} exiting lock", thread.currentthread.name);
}
finally
{
system.threading.monitor.exit(usingprinterlocker);
}
}
使用 lock 关键字通常比直接使用 monitor 类更可取,一方面是因为 lock 更简洁,另一方面是因为 lock 确保了即使受保护的代码引发异常,也可以释放基础监视器。这是通过 finally 关键字来实现的,无论是否引发异常它都执行关联的代码块。
4、同步事件和等待句柄
使用锁或监视器对于防止同时执行区分线程的代码块很有用,但是这些构造不允许一个线程向另一个线程传达事件。这需要“同步事件”,它是有两个状态(终止和非终止)的对象,可以用来激活和挂起线程。让线程等待非终止的同步事件可以将线程挂起,将事件状态更改为终止可以将线程激活。如果线程试图等待已经终止的事件,则线程将继续执行,而不会延迟。
同步事件有两种:autoresetevent和manualresetevent。它们之间唯一的不同在于,无论何时,只要autoresetevent激活线程,它的状态将自动从终止变为非终止。相反,manualresetevent允许它的终止状态激活任意多个线程,只有当它的reset方法被调用时才还原到非终止状态。
等待句柄,可以通过调用一种等待方法,如waitone、waitany或waitall,让线程等待事件。system.threading.waithandle.waitone使线程一直等待,直到单个事件变为终止状态;system.threading.waithandle.waitany阻止线程,直到一个或多个指示的事件变为终止状态;system.threading.waithandle.waitall阻止线程,直到所有指示的事件都变为终止状态。当调用事件的set方法时,事件将变为终止状态。
autoresetevent允许线程通过发信号互相通信。通常,当线程需要独占访问资源时使用该类。线程通过调用autoresetevent上的waitone来等待信号。如果autoresetevent为非终止状态,则线程会被阻止,并等待当前控制资源的线程通过调用set来通知资源可用。调用set向autoresetevent发信号以释放等待线程。autoresetevent将保持终止状态,直到一个正在等待的线程被释放,然后自动返回非终止状态。如果没有任何线程在等待,则状态将无限期地保持为终止状态。如果当autoresetevent为终止状态时线程调用waitone,则线程不会被阻止。autoresetevent将立即释放线程并返回到非终止状态。
可以通过将一个布尔值传递给构造函数来控制autoresetevent的初始状态:如果初始状态为终止状态,则为true;否则为false。
autoresetevent也可以同staticwaitall和waitany方法一起使用。
案例:
案例介绍:
今天我们来做饭,做饭呢,需要一菜、一粥。今天我们吃鱼。
熬粥和做鱼,是比较复杂的工作流程,
做粥:选材、淘米、熬制
做鱼:洗鱼、切鱼、腌制、烹调
为了提高效率,我们用两个线程来准备这顿饭,但是,现在只有一口锅,只能等一个做完之后,另一个才能进行最后的烹调。
来看实例代码:
usingsystem;
usingsystem.threading;
namespacemutithreadsample.threadsynchronization
{
///<summary>
///案例:做饭
///今天的dinner准备吃鱼,还要熬粥
///熬粥和做鱼,是比较复杂的工作流程,
///做粥:选材、淘米、熬制
///做鱼:洗鱼、切鱼、腌制、烹调
///我们用两个线程来准备这顿饭
///但是,现在只有一口锅,只能等一个做完之后,另一个才能进行最后的烹调
///</summary>
classcookresetevent
{
///<summary>
///
///</summary>
privateautoreseteventresetevent=newautoresetevent(false);
///<summary>
///做饭
///</summary>
publicvoidcook()
{
threadporridgethread=newthread(newthreadstart(porridge));
porridgethread.name="porridge";
porridgethread.start();
threadmakefishthread=newthread(newthreadstart(makefish));
makefishthread.name="makefish";
makefishthread.start();
//等待5秒
thread.sleep(5000);
resetevent.reset();
}
///<summary>
///熬粥
///</summary>
publicvoidporridge()
{
//选材
console.writeline("thread:{0},开始选材",thread.currentthread.name);
//淘米
console.writeline("thread:{0},开始淘米",thread.currentthread.name);
//熬制
console.writeline("thread:{0},开始熬制,需要2秒钟",thread.currentthread.name);
//需要2秒钟
thread.sleep(2000);
console.writeline("thread:{0},粥已经做好,锅闲了",thread.currentthread.name);
resetevent.set();
}
///<summary>
///做鱼
///</summary>
publicvoidmakefish()
{
//洗鱼
console.writeline("thread:{0},开始洗鱼",thread.currentthread.name);
//腌制
console.writeline("thread:{0},开始腌制",thread.currentthread.name);
//等待锅空闲出来
resetevent.waitone();
//烹调
console.writeline("thread:{0},终于有锅了",thread.currentthread.name);
console.writeline("thread:{0},开始做鱼,需要5秒钟",thread.currentthread.name);
thread.sleep(5000);
console.writeline("thread:{0},鱼做好了,好香",thread.currentthread.name);
resetevent.set();
}
}
}
manualresetevent与autoresetevent用法基本类似,这里不多做介绍。
5、mutex对象
mutex与监视器类似;它防止多个线程在某一时间同时执行某个代码块。事实上,名称“mutex”是术语“互相排斥(mutuallyexclusive)”的简写形式。然而与监视器不同的是,mutex可以用来使跨进程的线程同步。mutex由mutex类表示。当用于进程间同步时,mutex称为“命名mutex”,因为它将用于另一个应用程序,因此它不能通过全局变量或静态变量共享。必须给它指定一个名称,才能使两个应用程序访问同一个mutex对象。
尽管mutex可以用于进程内的线程同步,但是使用monitor通常更为可取,因为监视器是专门为.netframework而设计的,因而它可以更好地利用资源。相比之下,mutex类是win32构造的包装。尽管mutex比监视器更为强大,但是相对于monitor类,它所需要的互操作转换更消耗计算资源。
本地mutex和系统mutex
mutex分两种类型:本地mutex和命名系统mutex。如果使用接受名称的构造函数创建了mutex对象,那么该对象将与具有该名称的操作系统对象相关联。命名的系统mutex在整个操作系统中都可见,并且可用于同步进程活动。您可以创建多个mutex对象来表示同一命名系统mutex,而且您可以使用openexisting方法打开现有的命名系统mutex。
本地mutex仅存在于进程当中。进程中引用本地mutex对象的任意线程都可以使用本地mutex。每个mutex对象都是一个单独的本地mutex。
在本地mutex中,用法与monitor基本一致
继续修改前面的打印机案例:
声明mutex对象:
///<summary>
///mutex对象
///</summary>
privatestaticmutexmutex=newmutex();
具体操作:
///<summary>
///使用打印机进行打印
///</summary>
privatestaticvoiduseprinterwithmutex()
{
mutex.waitone();
try
{
console.writeline("{0}acquiredthelock",thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0}exitinglock",thread.currentthread.name);
}
finally
{
mutex.releasemutex();
}
}
多线程调用:
///<summary>
///测试
///</summary>
publicstaticvoidtestprint()
{
threadthread;
randomrandom=newrandom();
for(inti=0;i<computercount;i++)
{
thread=newthread(mythreadproc);
thread.name=string.format("thread{0}",i);
thread.sleep(random.next(3));
thread.start();
}
}
///<summary>
///线程执行操作
///</summary>
privatestaticvoidmythreadproc()
{
//使用打印机进行打印
//useprinter();
//monitor同步
//useprinterwithmonitor();
//用mutex同步
useprinterwithmutex();
//当前线程等待1秒
thread.sleep(1000);
}
最后的打印机案例代码:
usingsystem;
usingsystem.threading;
namespacemutithreadsample.threadsynchronization
{
classprinterwithlocktest
{
///<summary>
///正在使用的打印机
///</summary>
privatestaticobjectusingprinterlocker=newobject();
///<summary>
///计算机数量
///</summary>
publicstaticreadonlyintcomputercount=3;
///<summary>
///mutex对象
///</summary>
privatestaticmutexmutex=newmutex();
///<summary>
///测试
///</summary>
publicstaticvoidtestprint()
{
threadthread;
randomrandom=newrandom();
for(inti=0;i<computercount;i++)
{
thread=newthread(mythreadproc);
thread.name=string.format("thread{0}",i);
thread.sleep(random.next(3));
thread.start();
}
}
///<summary>
///线程执行操作
///</summary>
privatestaticvoidmythreadproc()
{
//使用打印机进行打印
//useprinter();
//monitor同步
//useprinterwithmonitor();
//用mutex同步
useprinterwithmutex();
//当前线程等待1秒
thread.sleep(1000);
}
///<summary>
///使用打印机进行打印
///</summary>
privatestaticvoiduseprinter()
{
//临界区
lock(usingprinterlocker)
{
console.writeline("{0}acquiredthelock",thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0}exitinglock",thread.currentthread.name);
}
}
///<summary>
///使用打印机进行打印
///</summary>
privatestaticvoiduseprinterwithmonitor()
{
system.threading.monitor.enter(usingprinterlocker);
try
{
console.writeline("{0}acquiredthelock",thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0}exitinglock",thread.currentthread.name);
}
finally
{
system.threading.monitor.exit(usingprinterlocker);
}
}
///<summary>
///使用打印机进行打印
///</summary>
privatestaticvoiduseprinterwithmutex()
{
mutex.waitone();
try
{
console.writeline("{0}acquiredthelock",thread.currentthread.name);
//模拟打印操作
thread.sleep(500);
console.writeline("{0}exitinglock",thread.currentthread.name);
}
finally
{
mutex.releasemutex();
}
}
}
}
6、读取器/编写器锁
readerwriterlockslim类允许多个线程同时读取一个资源,但在向该资源写入时要求线程等待以获得独占锁。
可以在应用程序中使用readerwriterlockslim,以便在访问一个共享资源的线程之间提供协调同步。获得的锁是针对readerwriterlockslim本身的。
设计您应用程序的结构,让读取和写入操作的时间尽可能最短。因为写入锁是排他的,所以长时间的写入操作会直接影响吞吐量。长时间的读取操作会阻止处于等待状态的编写器,并且,如果至少有一个线程在等待写入访问,则请求读取访问的线程也将被阻止。
案例:构造一个线程安全的缓存
usingsystem;
usingsystem.threading;
usingsystem.collections.generic;
namespacemutithreadsample.threadsynchronization
{
///<summary>
///同步cache
///</summary>
publicclasssynchronizedcache
{
privatereaderwriterlockslimcachelock=newreaderwriterlockslim();
privatedictionary<int,string>innercache=newdictionary<int,string>();
///<summary>
///读取
///</summary>
///<paramname="key"></param>
///<returns></returns>
publicstringread(intkey)
{
cachelock.enterreadlock();
try
{
returninnercache[key];
}
finally
{
cachelock.exitreadlock();
}
}
///<summary>
///添加项
///</summary>
///<paramname="key"></param>
///<paramname="value"></param>
publicvoidadd(intkey,stringvalue)
{
cachelock.enterwritelock();
try
{
innercache.add(key,value);
}
finally
{
cachelock.exitwritelock();
}
}
///<summary>
///添加项,有超时限制
///</summary>
///<paramname="key"></param>
///<paramname="value"></param>
///<paramname="timeout"></param>
///<returns></returns>
publicbooladdwithtimeout(intkey,stringvalue,inttimeout)
{
if(cachelock.tryenterwritelock(timeout))
{
try
{
innercache.add(key,value);
}
finally
{
cachelock.exitwritelock();
}
returntrue;
}
else
{
returnfalse;
}
}
///<summary>
///添加或者更新
///</summary>
///<paramname="key"></param>
///<paramname="value"></param>
///<returns></returns>
publicaddorupdatestatusaddorupdate(intkey,stringvalue)
{
cachelock.enterupgradeablereadlock();
try
{
stringresult=null;
if(innercache.trygetvalue(key,outresult))
{
if(result==value)
{
returnaddorupdatestatus.unchanged;
}
else
{
cachelock.enterwritelock();
try
{
innercache[key]=value;
}
finally
{
cachelock.exitwritelock();
}
returnaddorupdatestatus.updated;
}
}
else
{
cachelock.enterwritelock();
try
{
innercache.add(key,value);
}
finally
{
cachelock.exitwritelock();
}
returnaddorupdatestatus.added;
}
}
finally
{
cachelock.exitupgradeablereadlock();
}
}
///<summary>
///删除项
///</summary>
///<paramname="key"></param>
publicvoiddelete(intkey)
{
cachelock.enterwritelock();
try
{
innercache.remove(key);
}
finally
{
cachelock.exitwritelock();
}
}
///<summary>
///
///</summary>
publicenumaddorupdatestatus
{
added,
updated,
unchanged
};
}
}
7、semaphore和semaphoreslim
system.threading.semaphore类表示一个命名(系统范围)信号量或本地信号量。它是一个对win32信号量对象的精简包装。win32信号量是计数信号量,可用于控制对资源池的访问。
semaphoreslim类表示一个轻量的快速信号量,可用于在一个预计等待时间会非常短的进程内进行等待。semaphoreslim会尽可能多地依赖由公共语言运行时(clr)提供的同步基元。但是,它也会根据需要提供延迟初始化的、基于内核的等待句柄,以支持等待多个信号量。semaphoreslim还支持使用取消标记,但它不支持命名信号量或使用等待句柄来进行同步。
线程通过调用waitone方法来进入信号量,此方法是从waithandle类派生的。当调用返回时,信号量的计数将减少。当一个线程请求项而计数为零时,该线程会被阻止。当线程通过调用release方法释放信号量时,将允许被阻止的线程进入。并不保证被阻塞的线程进入信号量的顺序,例如先进先出(fifo)或后进先出(lifo)。信号量的计数在每次线程进入信号量时减小,在线程释放信号量时增加。当计数为零时,后面的请求将被阻塞,直到有其他线程释放信号量。当所有的线程都已释放信号量时,计数达到创建信号量时所指定的最大值。
案例分析:购买火车票
还得排队进行购买,购买窗口是有限的,只有窗口空闲时才能购买
usingsystem;
usingsystem.threading;
namespacemutithreadsample.threadsynchronization
{
///<summary>
///案例:支付流程
///如超市、药店、火车票等,都有限定的几个窗口进行结算,只有有窗口空闲,才能进行结算。
///我们就用多线程来模拟结算过程
///</summary>
classpaymentwithsemaphore
{
///<summary>
///声明收银员总数为3个,但是当前空闲的个数为0,可能还没开始上班。
///</summary>
privatestaticsemaphoreidlecashiers=newsemaphore(0,3);
///<summary>
///测试支付过程
///</summary>
publicstaticvoidtestpay()
{
parameterizedthreadstartstart=newparameterizedthreadstart(pay);
//假设同时有5个人来买票
for(inti=0;i<5;i++)
{
threadthread=newthread(start);
thread.start(i);
}
//主线程等待,让所有的的线程都激活
thread.sleep(1000);
//释放信号量,2个收银员开始上班了或者有两个空闲出来了
idlecashiers.release(2);
}
///<summary>
///
///</summary>
///<paramname="obj"></param>
publicstaticvoidpay(objectobj)
{
console.writeline("thread{0}beginsandwaitsforthesemaphore.",obj);
idlecashiers.waitone();
console.writeline("thread{0}startstopay.",obj);
//结算
thread.sleep(2000);
console.writeline("thread{0}:thepaymenthasbeenfinished.",obj);
console.writeline("thread{0}:releasethesemaphore.",obj);
idlecashiers.release();
}
}
}
8、障碍(barrier)4.0后技术
使多个任务能够采用并行方式依据某种算法在多个阶段中协同工作。
通过在一系列阶段间移动来协作完成一组任务,此时该组中的每个任务发信号指出它已经到达指定阶段的barrier并且暗中等待其他任务到达。相同的barrier可用于多个阶段。
9、spinlock(4.0后)
spinlock结构是一个低级别的互斥同步基元,它在等待获取锁时进行旋转。在多核计算机上,当等待时间预计较短且极少出现争用情况时,spinlock的性能将高于其他类型的锁。不过,我们建议您仅在通过分析确定system.threading.monitor方法或interlocked方法显著降低了程序的性能时使用spinlock。
即使spinlock未获取锁,它也会产生线程的时间片。它这样做是为了避免线程优先级别反转,并使垃圾回收器能够继续执行。在使用spinlock时,请确保任何线程持有锁的时间不会超过一个非常短的时间段,并确保任何线程在持有锁时不会阻塞。
由于spinlock是一个值类型,因此,如果您希望两个副本都引用同一个锁,则必须通过引用显式传递该锁。
usingsystem;
usingsystem.text;
usingsystem.threading;
usingsystem.threading.tasks;
namespacemutithreadsample.threadsynchronization
{
classspinlocksample
{
publicstaticvoidtest()
{
spinlockslock=newspinlock();
stringbuildersb=newstringbuilder();
actionaction=()=>
{
boolgotlock=false;
for(inti=0;i<100;i++)
{
gotlock=false;
try
{
slock.enter(refgotlock);
sb.append(i.tostring());
}
finally
{
//真正获取之后,才释放
if(gotlock)slock.exit();
}
}
};
//多线程调用action
parallel.invoke(action,action,action);
console.writeline("输出:{0}",sb.tostring());
}
}
}
10、spinwait(4.0后)
system.threading.spinwait是一个轻量同步类型,可以在低级别方案中使用它来避免内核事件所需的高开销的上下文切换和内核转换。在多核计算机上,当预计资源不会保留很长一段时间时,如果让等待线程以用户模式旋转数十或数百个周期,然后重新尝试获取资源,则效率会更高。如果在旋转后资源变为可用的,则可以节省数千个周期。如果资源仍然不可用,则只花费了少量周期,并且仍然可以进行基于内核的等待。这一旋转-等待的组合有时称为“两阶段等待操作”。
下面的基本示例采用微软案例:无锁堆栈
usingsystem;
usingsystem.threading;
namespacemutithreadsample.threadsynchronization
{
publicclasslockfreestack<t>
{
privatevolatilenodem_head;
privateclassnode{publicnodenext;publictvalue;}
publicvoidpush(titem)
{
varspin=newspinwait();
nodenode=newnode{value=item},head;
while(true)
{
head=m_head;
node.next=head;
if(interlocked.compareexchange(refm_head,node,head)==head)break;
spin.spinonce();
}
}
publicbooltrypop(outtresult)
{
result=default(t);
varspin=newspinwait();
nodehead;
while(true)
{
head=m_head;
if(head==null)returnfalse;
if(interlocked.compareexchange(refm_head,head.next,head)==head)
{
result=head.value;
returntrue;
}
spin.spinonce();
}
}
}
}
总结:
尽管有这么多的技术,但是不同的技术对应不同的场景,我们必须熟悉其特点和适用范围。在应用时,必须具体问题具体分析,选择最佳的同步方式。