首页 > 编程语言 >C#读写锁ReaderWriterLockSlim的使用

C#读写锁ReaderWriterLockSlim的使用

时间:2023-06-05 18:03:53浏览次数:38  
标签:Console Thread C# 读写 DateTime 线程 WriteLine GetHashCode ReaderWriterLockSlim

读写锁的概念很简单,允许多个线程同时获取读锁,但同一时间只允许一个线程获得写锁,因此也称作共享-独占锁。在C#中,推荐使用ReaderWriterLockSlim类来完成读写锁的功能。

某些场合下,对一个对象的读取次数远远大于修改次数,如果只是简单的用lock方式加锁,则会影响读取的效率。而如果采用读写锁,则多个线程可以同时读取该对象,只有等到对象被写入锁占用的时候,才会阻塞。

简单的说,当某个线程进入读取模式时,此时其他线程依然能进入读取模式,假设此时一个线程要进入写入模式,那么他不得不被阻塞。直到读取模式退出为止。

同样的,如果某个线程进入了写入模式,那么其他线程无论是要写入还是读取,都是会被阻塞的。

进入写入/读取模式有2种方法:

EnterReadLock尝试进入写入模式锁定状态。

TryEnterReadLock(Int32) 尝试进入读取模式锁定状态,可以选择整数超时时间。

EnterWriteLock 尝试进入写入模式锁定状态。

TryEnterWriteLock(Int32) 尝试进入写入模式锁定状态,可以选择超时时间。

退出写入/读取模式有2种方法:

ExitReadLock 减少读取模式的递归计数,并在生成的计数为 0(零)时退出读取模式。

ExitWriteLock 减少写入模式的递归计数,并在生成的计数为 0(零)时退出写入模式。

下面演示一下用法:

public class Program
  {
    static private ReaderWriterLockSlim rwl = new ReaderWriterLockSlim();
    static void Main(string[] args)
    {
      Thread t_read1 = new Thread(new ThreadStart(ReadSomething));
      t_read1.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read1.GetHashCode());
      Thread t_read2 = new Thread(new ThreadStart(ReadSomething));
      t_read2.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read2.GetHashCode());
      Thread t_write1 = new Thread(new ThreadStart(WriteSomething));
      t_write1.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write1.GetHashCode());
    }
    static public void ReadSomething()
    {
      Console.WriteLine("{0} Thread ID {1} Begin EnterReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      rwl.EnterReadLock();
      try
      {
        Console.WriteLine("{0} Thread ID {1} reading sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        Thread.Sleep(5000);//模拟读取信息
        Console.WriteLine("{0} Thread ID {1} reading end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
      finally
      {
        rwl.ExitReadLock();
        Console.WriteLine("{0} Thread ID {1} ExitReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
    }
    static public void WriteSomething()
    {
      Console.WriteLine("{0} Thread ID {1} Begin EnterWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      rwl.EnterWriteLock();
      try
      {
        Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        Thread.Sleep(10000);//模拟写入信息
        Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
      finally
      {
        rwl.ExitWriteLock();
        Console.WriteLine("{0} Thread ID {1} ExitWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
    }
  }

可以看到3号线程和4号线程能够同时进入读模式,而5号线程过了5秒钟后(即3,4号线程退出读锁后),才能进入写模式。

把上述代码修改一下,先开启2个写模式的线程,然后在开启读模式线程,代码如下:

static void Main(string[] args)
    {
      Thread t_write1 = new Thread(new ThreadStart(WriteSomething));
      t_write1.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write1.GetHashCode());
      Thread t_write2 = new Thread(new ThreadStart(WriteSomething));
      t_write2.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start WriteSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_write2.GetHashCode());
      Thread t_read1 = new Thread(new ThreadStart(ReadSomething));
      t_read1.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read1.GetHashCode());
      Thread t_read2 = new Thread(new ThreadStart(ReadSomething));
      t_read2.Start();
      Console.WriteLine("{0} Create Thread ID {1} , Start ReadSomething", DateTime.Now.ToString("hh:mm:ss fff"), t_read2.GetHashCode());
    }

结果如下:

可以看到,3号线程和4号线程都要进入写模式,但是3号线程先占用写入锁,因此4号线程不得不等了10s后才进入。5号线程和6号线程需要占用读取锁,因此等4号线程退出写入锁后才能继续下去。

TryEnterReadLock和TryEnterWriteLock可以设置一个超时时间,运行到这句话的时候,线程会阻塞在此,如果此时能占用锁,那么返回true,如果到超时时间还未占用锁,那么返回false,放弃锁的占用,直接继续执行下面的代码。

EnterUpgradeableReadLock

ReaderWriterLockSlim类提供了可升级读模式,这种方式和读模式的区别在于它还有通过调用 EnterWriteLock 或 TryEnterWriteLock 方法升级为写入模式。 因为每次只能有一个线程处于可升级模式。进入可升级模式的线程,不会影响读取模式的线程,即当一个线程进入可升级模式,任意数量线程可以同时进入读取模式,不会阻塞。如果有多个线程已经在等待获取写入锁,那么运行EnterUpgradeableReadLock将会阻塞,直到那些线程超时或者退出写入锁。

下面代码演示了如何在可升级读模式下,升级到写入锁。

static public void UpgradeableRead()
    {
      Console.WriteLine("{0} Thread ID {1} Begin EnterUpgradeableReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      rwl.EnterUpgradeableReadLock();
      try
      {
        Console.WriteLine("{0} Thread ID {1} doing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        Console.WriteLine("{0} Thread ID {1} Begin EnterWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        rwl.EnterWriteLock();
        try
        {
          Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
          Thread.Sleep(10000);//模拟写入信息
          Console.WriteLine("{0} Thread ID {1} writing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        }
        finally
        {
          rwl.ExitWriteLock();
          Console.WriteLine("{0} Thread ID {1} ExitWriteLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        }
        Thread.Sleep(10000);//模拟读取信息
        Console.WriteLine("{0} Thread ID {1} doing end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
      finally
      {
        rwl.ExitUpgradeableReadLock();
        Console.WriteLine("{0} Thread ID {1} ExitUpgradeableReadLock...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
    }

读写锁对于性能的影响是明显的。

下面测试代码:

public class Program
  {
    static private ReaderWriterLockSlim rwl = new ReaderWriterLockSlim();
    static void Main(string[] args)
    {
      Stopwatch sw = new Stopwatch();
      sw.Start();
      List<Task> lstTask = new List<Task>();
      for (int i = 0; i < 500; i++)
      {
        if (i % 25 != 0)
        {
          var t = Task.Factory.StartNew(ReadSomething);
          lstTask.Add(t);
        }
        else
        {
          var t = Task.Factory.StartNew(WriteSomething);
          lstTask.Add(t);
        }
      }
      Task.WaitAll(lstTask.ToArray());
      sw.Stop();
      Console.WriteLine("使用ReaderWriterLockSlim方式,耗时:" + sw.Elapsed);
      sw.Restart();
      lstTask = new List<Task>();
      for (int i = 0; i < 500; i++)
      {
        if (i % 25 != 0)
        {
          var t = Task.Factory.StartNew(ReadSomething_lock);
          lstTask.Add(t);
        }
        else
        {
          var t = Task.Factory.StartNew(WriteSomething_lock);
          lstTask.Add(t);
        }
      }
      Task.WaitAll(lstTask.ToArray());
      sw.Stop();
      Console.WriteLine("使用lock方式,耗时:" + sw.Elapsed);
    }
    static private object _lock1 = new object();
    static public void ReadSomething_lock()
    {
      lock (_lock1)
      {
        //Console.WriteLine("{0} Thread ID {1} reading sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
        Thread.Sleep(10);//模拟读取信息
        //Console.WriteLine("{0} Thread ID {1} reading end.", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());
      }
    }
    static public void WriteSomething_lock()
    {
      lock (_lock1)
      {
        //Console.WriteLine("{0} Thread ID {1} writing sth...", DateTime.Now.ToString("hh:mm:ss fff"), Thread.CurrentThread.GetHashCode());

标签:Console,Thread,C#,读写,DateTime,线程,WriteLine,GetHashCode,ReaderWriterLockSlim
From: https://blog.51cto.com/u_4018548/6418211

相关文章

  • C++11中的std::function
    看看这段代码先来看看下面这两行代码:std::function<void(EventKeyboard::KeyCode,Event*)>onKeyPressed;std::function<void(EventKeyboard::KeyCode,Event*)>onKeyReleased;这两行代码是从Cocos2d-x中摘出来的,重点是这两行代码的定义啊。std::function这是什么东西?如果你对上......
  • ChatJPT:开创人机交互新纪元的技术突破
      人工智能技术的快速发展正在深刻改变着我们与机器之间的交互方式。ChatJPT作为一项创新的技术,为人机交互带来了全新的可能性。本文将探讨ChatJPT的技术原理、应用场景以及对未来社会的影响。ChatJPT的技术原理ChatJPT是基于GPT-3.5架构开发的大型语言模型,它具备深度学习和自......
  • 使用 TypeScript 探索面向对象编程
    在软件开发领域,面向对象编程(OOP)已成为创建复杂且可扩展应用程序的基本范例。支持OOP概念的最流行的编程语言之一是TypeScript。TypeScript是JavaScript的超集,它增加了静态类型和其他功能以增强代码的可维护性和可读性。在这篇博客中,我们将探讨TypeScript中面向对象编程......
  • Q370qC化学成分、Q370qC钢板力学性能、Q370qC钢板期货订轧
    一、Q370qC钢板简介:Q370qC属于桥梁用钢板,Q370qC钢板不仅有非常高的强度钢还有足够的韧性,多数应用在铆接及栓焊结构的公路桥及铁路桥梁结构等应用非常之广泛也是搭建桥梁的不可缺失的材料之一。Q370qC桥梁板执行标准为;GB/T714-2014.Q370qC钢板牌号表示:“Q”:表示钢的屈服强度的“屈”......
  • 使用 python-fire 快速构建 CLI
    命令行应用程序是开发人员最好的朋友。想快速完成某事?只需敲击几下键盘,您就已经拥有了想要的东西。Python是许多开发人员在需要快速组合某些东西时选择的第一语言。但是我们拼凑起来的东西在大多数时候并不是一个完整的CLI,您需要管理标志、解析参数、链接子命令等等,这很麻烦,因此......
  • C# new 和override重写的区别
    在C#中,函数前面加override和new都可以实现函数的重写(Overriding)。不过它们实现的方式不同,因此会有一些区别。1.Override在C#中,override关键字主要用于重写父类中虚方法(VirtualMethod),它表示子类中的方法会覆盖父类中的同名方法。使用override关键字后,子类的方法必须......
  • 前端实现导出word文档docx格式
    说明前端实现导出word文档,我们需要用到docxtemplater这个库使用的是vue2.6和vue-cli5还需要准备一个word模板,更多模板变量请去docxtemplater官网获取准备word模板安装需要用到的库//安装docxtemplaternpminstalldocxtemplaterpizzip--save//安装jszip-utilsn......
  • AH8669_交流AC220V降压转5V 300MA左右非隔离电源方案
    AH8669是一颗低成本的非隔离开关高性能交流转直流的转换器降压芯片,内部集成650V高耐压功率MOSFET额定700MA电流输出,非常适应于消费类的小家电控制模块以及给MCU供电和智能插座的家用电器上,交流转12V外围元件少,电路简单,内部集成软启动电路具有多功能保护有过载保护、过压保护、......
  • 嵌入式知识分享:Docker容器部署方法说明
    前 言本指导文档适用开发环境:Windows开发环境:Windows764bit、Windows1064bitLinux开发环境:Ubuntu18.04.464bit虚拟机:VMware15.1.0Docker是一个开源的应用容器引擎,让开发者可打包他们的应用以及依赖包到一个可移植的镜像中,然后发布到任何流行的Linux或Windows机器上,亦......
  • 【解决方法】网络设备使用CLI命令行模式进入SSH登录,如交换机,路由器
    环境:工具:锐捷EVE模拟器,VMwareWorkstationPro远程工具:SecureCRT系统版本:Windows10问题描述:描述:在实验使用路由器远程SSH登录交换机时,无法连接,一直提示:%Unrecognizedhostoraddress,orprotocolnotrunning.提示:若按照教程还是无法完成操作,可以进入右侧的企鹅,......