C#线程系列讲座(5):同步技术之Monitor

简介:

本文为原创,如需转载,请注明作者和出处,谢谢!

上一篇:C#线程系列讲座(4):同步与死锁

    在上一讲介绍了使用lock来实现线程之间的同步。实际上,这个lockC#的一个障眼法,在C#编译器编译lock语句时,将其编译成了调用Monitor类。先看看下面的C#源代码:


复制代码
public   static   void  MyLock()
{
    
lock  ( typeof (Program))
    {
    }
}
复制代码


    上面的代码通过lock语句使MyLock同步,这个方法被编译成IL后,代码如图1所示。



                                                         图1

    从上图被标注的区域可以看到,一条lock语句被编译成了调用MonitorEnterExit方法。MonitorSystem.Threading命名空间中。lock的功能就相当于直接调用MonitorEntry方法,所不同的是,lock方法在结束后,会自动解除锁定,当然,在IL中是调用了MonitorExit方法,但在C#程序中,看起来是自动解锁的,这类似于C#中的using语句,可以自动释放数据库等的资源。但如果直接在C#源程序中使用Monitor类,就必须调用Exit方法来显式地解除锁定。如下面的代码所示:

复制代码
Monitor.Entry(lockObj);
try
{
    
//  lockObj的同布区
}
catch (Exception e)
{
    
//  异常处理代码
}
finally
{
    Monitor.Exit(lockObj);  
//  解除锁定
}
复制代码


    Exit方法最后在finally里调用,这样无论在方法在发生异常、返回还是正常执行,都会执行到finally,并调用Exit方法解除锁定。

    Monitor类不仅可以完全取代lock语句(如果只使用lock语句本身的功能,最好还是直接用lock语句吧),还可以使用TryEntry方法设置一个锁定超时,单位是毫秒。如下面的代码所示:

复制代码
if (Monitor.TryEntry(lockObj,  1000 ))
{
    
try
    {
    }
    
finally
    {
        Monitor.Exit(lockObj);
    }
}
else
{
    
//  超时后的处理代码
}
复制代码

    上面的代码设置了锁定超时时间为1秒,也就是说,在1秒中后,lockObj还未被解锁,TryEntry方法就会返回false,如果在1秒之内,lockObj被解锁,TryEntry返回true。我们可以使用这种方法来避免死锁,如下面的代码所示:

复制代码
     class  Program
    {
        
private   static  Object objA  =   new  Object();
        
private   static  Object objB  =   new  Object();
        
public   static   void  LockA()
        {
            
if  (Monitor.TryEnter(objA,  1000 ))
            {
                Thread.Sleep(
1000 );
                
if  (Monitor.TryEnter(objB,  2000 ))
                {
                    Monitor.Exit(objB);
                }
                
else
                {

                    Console.WriteLine(
" LockB timeout " );
                }
                Monitor.Exit(objA);
            }
            Console.WriteLine(
" LockA " );
        }
        
public   static   void  LockB()
        {
            
if  (Monitor.TryEnter(objB,  2000 ))
            {
                Thread.Sleep(
2000 );
                
if  (Monitor.TryEnter(objA,  1000 ))
                {
                    Monitor.Exit(objA);
                }
                
else
                {
                    Console.WriteLine(
" LockA timeout " );
                }
                Monitor.Exit(objB);
            }
            Console.WriteLine(
" LockB " );
        }
        
public   static   void  Main()
        {
            Thread threadA 
=   new  Thread(LockA);
            Thread threadB 
=   new  Thread(LockB);
            threadA.Start();
            threadB.Start();
            Thread.Sleep(
4000 );         
            Console.WriteLine(
" 线程结束 " );
        }
    }
复制代码

    上面的代码是在上一讲举的死锁的例子,但在这一讲将lock语句改成了TryEntry方法,而且设置了锁定超时间,由于在等待一定时间后,不管被锁定的对象是否被解锁,TryEntry方法都会返回,因此,上面的代码是不会死锁的。运行上面的代码的结果如图2所示。



                                              图2

    如果TryEntry方法的超时时间为System.Threading.Timeout.InfiniteTryEntry方法就相当于Entry方法,如果超时时间为0,不管是否解锁,TryEntry方法都会立即返回。

本文转自银河使者博客园博客,原文链接 http://www.cnblogs.com/nokiaguy/archive/2008/07/31/1257625.html如需转载请自行联系原作者

银河使者
相关文章
|
4天前
|
Java 测试技术 开发工具
Android 笔记:AndroidTrain , Lint , build(1),只需一篇文章吃透Android多线程技术
Android 笔记:AndroidTrain , Lint , build(1),只需一篇文章吃透Android多线程技术
|
4天前
|
关系型数据库 MySQL Java
实时计算 Flink版产品使用合集之mysql通过flink cdc同步数据,有没有办法所有表共用一个dump线程
实时计算Flink版作为一种强大的流处理和批处理统一的计算框架,广泛应用于各种需要实时数据处理和分析的场景。实时计算Flink版通常结合SQL接口、DataStream API、以及与上下游数据源和存储系统的丰富连接器,提供了一套全面的解决方案,以应对各种实时计算需求。其低延迟、高吞吐、容错性强的特点,使其成为众多企业和组织实时数据处理首选的技术平台。以下是实时计算Flink版的一些典型使用合集。
9 0
|
6天前
|
安全 C++
C++多线程编程:并发与同步
C++多线程编程:并发与同步
12 0
|
6天前
|
C#
C#同步异步详解
C#同步异步详解
9 0
|
6天前
|
安全 Java C#
C#多线程详解
C#多线程详解
14 0
|
6天前
|
存储 缓存 前端开发
Java串口通信技术探究3:RXTX库线程 优化系统性能的SerialPortEventListener类
Java串口通信技术探究3:RXTX库线程 优化系统性能的SerialPortEventListener类
60 3
|
6天前
|
Dart 前端开发 安全
【Flutter前端技术开发专栏】Flutter中的线程与并发编程实践
【4月更文挑战第30天】本文探讨了Flutter中线程管理和并发编程的关键性,强调其对应用性能和用户体验的影响。Dart语言提供了`async`、`await`、`Stream`和`Future`等原生异步支持。Flutter采用事件驱动的单线程模型,通过`Isolate`实现线程隔离。实践中,可利用`async/await`、`StreamBuilder`和`Isolate`处理异步任务,同时注意线程安全和性能调优。参考文献包括Dart异步编程、Flutter线程模型和DevTools文档。
【Flutter前端技术开发专栏】Flutter中的线程与并发编程实践
|
6天前
|
存储 安全 Java
【亮剑】`ConcurrentHashMap`是Java中线程安全的哈希表,采用锁定分离技术提高并发性能
【4月更文挑战第30天】`ConcurrentHashMap`是Java中线程安全的哈希表,采用锁定分离技术提高并发性能。数据被分割成多个Segment,每个拥有独立锁,允许多线程并发访问不同Segment。当写操作发生时,计算键的哈希值定位Segment并获取其锁;读操作通常无需锁定。内部会根据负载动态调整Segment,减少锁竞争。虽然使用不公平锁,但Java 8及以上版本提供了公平锁选项。理解其工作原理对开发高性能并发应用至关重要。
|
6天前
|
缓存 Java 编译器
第一章 Java线程池技术应用
第一章 Java线程池技术应用
19 0
|
6天前
|
并行计算 算法 安全
Java从入门到精通:2.1.3深入学习Java核心技术——掌握Java多线程编程
Java从入门到精通:2.1.3深入学习Java核心技术——掌握Java多线程编程