本篇的内容主要是介绍 ReaderWriterLockSlim 类,来实现多线程下的读写分离。

ReaderWriterLockSlim

ReaderWriterLock 类:定义支持单个写线程和多个读线程的锁。

ReaderWriterLockSlim 类:表示用于管理资源访问的锁定状态,可实现多线程读取或进行独占式写入访问。

两者的 API 十分接近,而且 ReaderWriterLockSlim 相对 ReaderWriterLock 来说 更加安全。因此本文主要讲解 ReaderWriterLockSlim 。

两者都是实现多个线程可同时读取、只允许一个线程写入的类。

ReaderWriterLockSlim

老规矩,先大概了解一下 ReaderWriterLockSlim 常用的方法。

常用方法

方法 说明
EnterReadLock() 尝试进入读取模式锁定状态。
EnterUpgradeableReadLock() 尝试进入可升级模式锁定状态。
EnterWriteLock() 尝试进入写入模式锁定状态。
ExitReadLock() 减少读取模式的递归计数,并在生成的计数为 0(零)时退出读取模式。
ExitUpgradeableReadLock() 减少可升级模式的递归计数,并在生成的计数为 0(零)时退出可升级模式。
ExitWriteLock() 减少写入模式的递归计数,并在生成的计数为 0(零)时退出写入模式。
TryEnterReadLock(Int32) 尝试进入读取模式锁定状态,可以选择整数超时时间。
TryEnterReadLock(TimeSpan) 尝试进入读取模式锁定状态,可以选择超时时间。
TryEnterUpgradeableReadLock(Int32) 尝试进入可升级模式锁定状态,可以选择超时时间。
TryEnterUpgradeableReadLock(TimeSpan) 尝试进入可升级模式锁定状态,可以选择超时时间。
TryEnterWriteLock(Int32) 尝试进入写入模式锁定状态,可以选择超时时间。
TryEnterWriteLock(TimeSpan) 尝试进入写入模式锁定状态,可以选择超时时间。

ReaderWriterLockSlim 的读、写入锁模板如下:

  1. private static ReaderWriterLockSlim toolLock = new ReaderWriterLockSlim();
  2. // 读
  3. private T Read()
  4. {
  5. try
  6. {
  7. toolLock.EnterReadLock(); // 获取读取锁
  8. return obj;
  9. }
  10. catch { }
  11. finally
  12. {
  13. toolLock.ExitReadLock(); // 释放读取锁
  14. }
  15. return default;
  16. }
  17. // 写
  18. public void Write(int key, int value)
  19. {
  20. try
  21. {
  22. toolLock.EnterUpgradeableReadLock();
  23. try
  24. {
  25. toolLock.EnterWriteLock();
  26. /*
  27. *
  28. */
  29. }
  30. catch
  31. {
  32. }
  33. finally
  34. {
  35. toolLock.ExitWriteLock();
  36. }
  37. }
  38. catch { }
  39. finally
  40. {
  41. toolLock.ExitUpgradeableReadLock();
  42. }
  43. }

订单系统示例

这里来模拟一个简单粗糙的订单系统。

开始编写代码前,先来了解一些方法的具体使用。

EnterReadLock() / TryEnterReadLockExitReadLock() 成对出现。

EnterWriteLock() / TryEnterWriteLock()ExitWriteLock() 成对出现。

EnterUpgradeableReadLock() 进入可升级的读模式锁定状态。

EnterReadLock() 使用 EnterUpgradeableReadLock() 进入升级状态,在恰当时间点 通过 EnterWriteLock() 进入写模式。(也可以倒过来)

定义三个变量:

ReaderWriterLockSlim 多线程读写锁;

MaxId 当前订单 Id 的最大值;

orders 订单表;

  1. private static ReaderWriterLockSlim tool = new ReaderWriterLockSlim(); // 读写锁
  2. private static int MaxId = 1;
  3. public static List<DoWorkModel> orders = new List<DoWorkModel>(); // 订单表
  1. // 订单模型
  2. public class DoWorkModel
  3. {
  4. public int Id { get; set; } // 订单号
  5. public string UserName { get; set; } // 客户名称
  6. public DateTime DateTime { get; set; } // 创建时间
  7. }

然后实现查询和创建订单的两个方法。

分页查询订单:

在读取前使用 EnterReadLock() 获取锁;

读取完毕后,使用 ExitReadLock() 释放锁。

这样能够在多线程环境下保证每次读取都是最新的值。

  1. // 分页查询订单
  2. private static DoWorkModel[] DoSelect(int pageNo, int pageSize)
  3. {
  4. try
  5. {
  6. DoWorkModel[] doWorks;
  7. tool.EnterReadLock(); // 获取读取锁
  8. doWorks = orders.Skip((pageNo - 1) * pageSize).Take(pageSize).ToArray();
  9. return doWorks;
  10. }
  11. catch { }
  12. finally
  13. {
  14. tool.ExitReadLock(); // 释放读取锁
  15. }
  16. return default;
  17. }

创建订单:

创建订单的信息十分简单,知道用户名和创建时间就行。

订单系统要保证的时每个 Id 都是唯一的(实际情况应该用Guid),这里为了演示读写锁,设置为 数字。

在多线程环境下,我们不使用 Interlocked.Increment() ,而是直接使用 += 1,因为有读写锁的存在,所以操作也是原则性的。

  1. // 创建订单
  2. private static DoWorkModel DoCreate(string userName, DateTime time)
  3. {
  4. try
  5. {
  6. tool.EnterUpgradeableReadLock(); // 升级
  7. try
  8. {
  9. tool.EnterWriteLock(); // 获取写入锁
  10. // 写入订单
  11. MaxId += 1; // Interlocked.Increment(ref MaxId);
  12. DoWorkModel model = new DoWorkModel
  13. {
  14. Id = MaxId,
  15. UserName = userName,
  16. DateTime = time
  17. };
  18. orders.Add(model);
  19. return model;
  20. }
  21. catch { }
  22. finally
  23. {
  24. tool.ExitWriteLock(); // 释放写入锁
  25. }
  26. }
  27. catch { }
  28. finally
  29. {
  30. tool.ExitUpgradeableReadLock(); // 降级
  31. }
  32. return default;
  33. }

Main 方法中:

开 5 个线程,不断地读,开 2 个线程不断地创建订单。线程创建订单时是没有设置 Thread.Sleep() 的,因此运行速度十分快。

Main 方法里面的代码没有什么意义。

  1. static void Main(string[] args)
  2. {
  3. // 5个线程读
  4. for (int i = 0; i < 5; i++)
  5. {
  6. new Thread(() =>
  7. {
  8. while (true)
  9. {
  10. var result = DoSelect(1, MaxId);
  11. if (result is null)
  12. {
  13. Console.WriteLine("获取失败");
  14. continue;
  15. }
  16. foreach (var item in result)
  17. {
  18. Console.Write($"{item.Id}|");
  19. }
  20. Console.WriteLine("\n");
  21. Thread.Sleep(1000);
  22. }
  23. }).Start();
  24. }
  25. for (int i = 0; i < 2; i++)
  26. {
  27. new Thread(() =>
  28. {
  29. while(true)
  30. {
  31. var result = DoCreate((new Random().Next(0, 100)).ToString(), DateTime.Now); // 模拟生成订单
  32. if (result is null)
  33. Console.WriteLine("创建失败");
  34. else Console.WriteLine("创建成功");
  35. }
  36. }).Start();
  37. }
  38. }

在 ASP.NET Core 中,则可以利用读写锁,解决多用户同时发送 HTTP 请求带来的数据库读写问题。

这里就不做示例了。

如果另一个线程发生问题,导致迟迟不能交出写入锁,那么可能会导致其它线程无限等待。

那么可以使用 TryEnterWriteLock() 并且设置等待时间,避免阻塞时间过长。

  1. bool isGet = tool.TryEnterWriteLock(500);

并发字典写示例

因为理论的东西,笔者这里不会说太多,主要就是先掌握一些 API(方法、属性) 的使用,然后简单写出示例,后面再慢慢深入了解底层原理。

这里来写一个多线程共享使用字典(Dictionary)的使用示例。

增加两个静态变量:

  1. private static ReaderWriterLockSlim toolLock = new ReaderWriterLockSlim();
  2. private static Dictionary<int, int> dict = new Dictionary<int, int>();

实现一个写操作:

  1. public static void Write(int key, int value)
  2. {
  3. try
  4. {
  5. // 升级状态
  6. toolLock.EnterUpgradeableReadLock();
  7. // 读,检查是否存在
  8. if (dict.ContainsKey(key))
  9. return;
  10. try
  11. {
  12. // 进入写状态
  13. toolLock.EnterWriteLock();
  14. dict.Add(key,value);
  15. }
  16. finally
  17. {
  18. toolLock.ExitWriteLock();
  19. }
  20. }
  21. finally
  22. {
  23. toolLock.ExitUpgradeableReadLock();
  24. }
  25. }

上面没有 catch { } 是为了更好观察代码,因为使用了读写锁,理论上不应该出现问题的。

模拟五个线程同时写入字典,由于不是原子操作,所以 sum 的值有些时候会出现重复值。

原子操作请参考:https://www.cnblogs.com/whuanle/p/12724371.html#1,出现问题

  1. private static int sum = 0;
  2. public static void AddOne()
  3. {
  4. for (int i = 0; i < 100_0000; i++)
  5. {
  6. sum += 1;
  7. Write(sum,sum);
  8. }
  9. }
  10. static void Main(string[] args)
  11. {
  12. for (int i = 0; i < 5; i++)
  13. new Thread(() => { AddOne(); }).Start();
  14. Console.ReadKey();
  15. }

ReaderWriterLock

大多数情况下都是推荐 ReaderWriterLockSlim 的,而且两者的使用方法十分接近。

例如 AcquireReaderLock 是获取读锁,AcquireWriterLock 获取写锁。使用对应的方法即可替换 ReaderWriterLockSlim 中的示例。

这里就不对 ReaderWriterLock 进行赘述了。

ReaderWriterLock 的常用方法如下:

方法 说明
AcquireReaderLock(Int32) 使用一个 Int32 超时值获取读线程锁。
AcquireReaderLock(TimeSpan) 使用一个 TimeSpan 超时值获取读线程锁。
AcquireWriterLock(Int32) 使用一个 Int32 超时值获取写线程锁。
AcquireWriterLock(TimeSpan) 使用一个 TimeSpan 超时值获取写线程锁。
AnyWritersSince(Int32) 指示获取序列号之后是否已将写线程锁授予某个线程。
DowngradeFromWriterLock(LockCookie) 将线程的锁状态还原为调用 UpgradeToWriterLock(Int32) 前的状态。
ReleaseLock() 释放锁,不管线程获取锁的次数如何。
ReleaseReaderLock() 减少锁计数。
ReleaseWriterLock() 减少写线程锁上的锁计数。
RestoreLock(LockCookie) 将线程的锁状态还原为调用 ReleaseLock() 前的状态。
UpgradeToWriterLock(Int32) 使用一个 Int32 超时值将读线程锁升级为写线程锁。
UpgradeToWriterLock(TimeSpan) 使用一个 TimeSpan 超时值将读线程锁升级为写线程锁。

官方示例可以看:

https://docs.microsoft.com/zh-cn/dotnet/api/system.threading.readerwriterlock?view=netcore-3.1#examples

C#多线程(10):读写锁的更多相关文章

  1. Java多线程之读写锁机制

    Java多线程中有很多的锁机制,他们都有各自的应用场景,例如今天我说的这种锁机制:读写锁 读写锁,见名知意,主要可以进行两种操作,读和写操作,他们之间结合使用起来又是各不相同的.比如多个线程之间可以同 ...

  2. 技术笔记:Delphi多线程应用读写锁

    在多线程应用中锁是一个很简单又很复杂的技术,之所以要用到锁是因为在多进程/线程环境下,一段代码可能会被同时访问到,如果这段代码涉及到了共享资源(数据)就需要保证数据的正确性.也就是所谓的线程安全.之前 ...

  3. java多线程 -- ReadWriteLock 读写锁

    写一条线程,读多条线程能够提升效率. 写写/读写 需要“互斥”;读读 不需要互斥. ReadWriteLock 维护了一对相关的锁,一个用于只读操作,另一个用于写入操作.只要没有 writer,读取锁 ...

  4. java 多线程 day12 读写锁

    import java.util.Random;import java.util.concurrent.locks.ReadWriteLock;import java.util.concurrent. ...

  5. 用读写锁三句代码解决多线程并发写入文件 z

    C#使用读写锁三句代码简单解决多线程并发写入文件时提示“文件正在由另一进程使用,因此该进程无法访问此文件”的问题 在开发程序的过程中,难免少不了写入错误日志这个关键功能.实现这个功能,可以选择使用第三 ...

  6. C# 防止同时调用=========使用读写锁三行代码简单解决多线程并发的问题

    http://www.jb51.net/article/99718.htm     本文主要介绍了C#使用读写锁三行代码简单解决多线程并发写入文件时提示"文件正在由另一进程使用,因此该进程无 ...

  7. C#使用读写锁三行代码简单解决多线程并发写入文件时线程同步的问题

    (补充:初始化FileStream时使用包含文件共享属性(System.IO.FileShare)的构造函数比使用自定义线程锁更为安全和高效,更多内容可点击参阅) 在开发程序的过程中,难免少不了写入错 ...

  8. C#使用读写锁解决多线程并发写入文件时线程同步的问题

    读写锁是以 ReaderWriterLockSlim 对象作为锁管理资源的,不同的 ReaderWriterLockSlim 对象中锁定同一个文件也会被视为不同的锁进行管理,这种差异可能会再次导致文件 ...

  9. c++ 读写锁

    #ifndef THREAD_UTIL_H #define THREAD_UTIL_H #include <pthread.h> namespace spider { class Auto ...

  10. Java多线程13:读写锁和两种同步方式的对比

    读写锁ReentrantReadWriteLock概述 大型网站中很重要的一块内容就是数据的读写,ReentrantLock虽然具有完全互斥排他的效果(即同一时间只有一个线程正在执行lock后面的任务 ...

随机推荐

  1. element-ui中Select 选择器列表内容居中

    <el-select class="my-el-select" v-model="tenantCont" placeholder="请输入机构标 ...

  2. 让一段代码执行在new Vue之前

    这是一个自调用函数,也有人叫做一次性函数: 这样函数前面最后打一个: ;(function initApp(){ loadApp(); })() function loadApp (){ //tena ...

  3. 【小实验】golang中的字节对齐

    作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 使用golang来调用SIMD指令,发现程序崩溃了: __ ...

  4. 【JS 逆向百例】无限debugger绕过,某政民互动数据逆向

    声明 本文章中所有内容仅供学习交流,抓包内容.敏感网址.数据接口均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请联系我立即删除! 逆向目标 目标:某政务服务 ...

  5. git push卡住了 git push writing object

    解决方案: 执行命令:$ git config --global http.postBuffer 524288000 再次push 会卡在这里:POST git-receive-pack(892384 ...

  6. 用户 'NT Service\SSISScaleOutMaster140' 登录失败

    用户 'NT Service\SSISScaleOutMaster140' 登录失败. 原因: 找不到与提供的名称匹配的登录名. 项目情况: 用户 'NT Service\SSISScaleOutMa ...

  7. Abp 模板更换数据库 版本为V5.x,遇到的问题

    数据库的选择: Mysql使用5.0.0的版本,根据在下面的依赖项 大于等于5.0.5 && 小于6.0.0 Microsoft.EntityFrameworkCore.Tools和M ...

  8. 【五】强化学习之Sarsa、Qlearing详细讲解----PaddlePaddlle【PARL】框架{飞桨}

    相关文章: [一]飞桨paddle[GPU.CPU]安装以及环境配置+python入门教学 [二]-Parl基础命令 [三]-Notebook.&pdb.ipdb 调试 [四]-强化学习入门简 ...

  9. <semaphore.h> 和 <sys/sem.h> 的区别

    <sys/sem.h>为 XSI(最初是 Unix System V)信号量提供接口. 这些不是基本 POSIX 标准的一部分(它们在 XSI 选项中,主要是为了传统的 Unix 兼容性) ...

  10. NC16416 [NOIP2017]逛公园

    题目链接 题目 题目描述 策策同学特别喜欢逛公园. 公园可以看成一张 N 个点 M 条边构成的有向图,且没有自环和重边.其中 1 号点是公园的入口, N 号点是公园的出口,每条边有一个非负权值,代表策 ...