当前位置: 首页 > news >正文

C#读写锁与并发控制

前言

我们在使用一些资源的时候,有些资源可能是临界资源,就是同一时刻只能允许一个进程进行访问,比如打印机,或者文件的一些写入操作,我们知道,当我们有多个进程同时往同一个文件写入内容的时候,肯定出实现如下的一个错误。

The process cannot access the file ‘D:\MyProject\TestProject\并发测试\bin\Debug\net8.0\test.txt’ because it is being used by another process.

所以写入文件的那一段代码就是属于临界资源,但是在我们的程序中,现在都推荐使用异步操作,那我们在使用Task等异步方法的时候,有些程序段可能会在其他的线程进行处理,所以有些时候就不可避免的产生并发的问题,对于文件写入的操作,系统会直接报错,对于数据库写入或者更新的操作,可能就会产生死锁问题。当然解决这些问题并没有一个统一的解决方式,可以加锁,使用队列,数据库用主从复制,读写分离等方法来优化,那么本文主要来实验一下C#中读写锁的控制,了解一些原理,可能对我们解决问题有所帮助。

需求

现在我们有一个文本文件,可能会进行多线程的读和写操作,如何不报错,实现多线程的并发控制问题。

实战

这里还是新建一个控制台项目,然后新建一个名为test.txt的文本文件,设置属性复制到输出目录为始终复制。
接着新建四个方法,两个写操作,两个读操作。代码如下。

public static void WriteTxt1()
{File.WriteAllText("test.txt", "hello 111");
}
public static void WriteTxt2()
{File.WriteAllText("test.txt", "hello 222");
}
public static void ReadTxt1()
{var fileInfo = File.ReadAllText("test.txt");Console.WriteLine($"ReadTxt1读取内容:{fileInfo}");
}
public static void ReadTxt1()
{var fileInfo = File.ReadAllText("test.txt");Console.WriteLine($"ReadTxt2读取内容:{fileInfo}");
}

这里没什么好说的,就是非常简单的读取和写入文件的操作。然后在Main方法里添加如下代码。

static void Main(string[] args)
{Parallel.Invoke(WriteTxt1, WriteTxt2);Console.WriteLine("****数据查询完成****");Console.ReadKey();
}

这里使用Parallel来实现多线程的并发操作。就这样运行的话,不出意外会提示开头的错误,不允许多个进程同时写入文件。
所以我们新增一个变量。

private static ReaderWriterLockSlim lockSlim = new ReaderWriterLockSlim();

然后写入操作修改如下。

 public static void WriteTxt1(){lockSlim.EnterWriteLock();try{File.WriteAllText("test.txt", "hello 111");}finally{lockSlim.ExitWriteLock();}}public static void WriteTxt2(){lockSlim.EnterWriteLock();try{File.WriteAllText("test.txt", "hello 222");}finally{lockSlim.ExitWriteLock();}}

主要是增加了写锁和释放写锁。
读取操作就可以新增和释放读锁。代码如下。

 public static void ReadTxt1(){lockSlim.EnterReadLock();try{var fileInfo = File.ReadAllText("test.txt");Console.WriteLine($"ReadTxt1读取内容:{fileInfo}");}finally{lockSlim.ExitReadLock();}}public static void ReadTxt2(){lockSlim.EnterReadLock();try{var fileInfo = File.ReadAllText("test.txt");Console.WriteLine($"ReadTxt2读取内容:{fileInfo}");}finally{lockSlim.ExitReadLock();}}

总结

调试的时候可以使用Thread.Sleep(3000);来辅助调试,通过调试发现,如果在两个写入方法里都添加写锁,那么只会有一个线程可以继续往下执行,另一个线程的写操作,则一直停留在添加写锁的地方,当该资源区没有写锁的时候(其他写锁已释放),那么才可以添加新的写锁。这样就保证了临界资源的线性使用。

通过调试还可以发现以下规则。

  • 先加写锁,则不能在加读锁和写锁。
  • 先加读锁,还可以在加读锁,但不能加写锁。

结语

感谢您花时间阅读这篇文章,希望这些知识能为您的技术之旅增添一份力量。明天会更好!

Study hard and make progress every day.

欢迎关注下方微信公众号,一起学习,一起娱乐,一起进步,点击卡片可以查看公众号二维码哟。


http://www.mrgr.cn/news/21128.html

相关文章:

  • 电脑桌面整理怎么弄?分享8款桌面整理软件,轻松拿捏桌面美化!
  • 【脚手架 第一篇章】介绍一下若依微服务版框架
  • 超级兔子与这三款恢复工具:性能对比与用户体验分析
  • 数学基础 -- 线性代数之矩阵正定性
  • 10款古方突破1800亿元,康缘药业发力,市场迎井喷式增长……
  • 2024.9.6 作业
  • 【简历】25届南京某一本JAVA简历:简历通过率还好,但是拿不到OFFER
  • Qt-布局管理
  • 太极八卦图时钟屏保 这个电脑屏保很有个性哦 你喜欢吗?
  • timm从本地加载预训练模型
  • Jmeter模拟用户登录时获取token如何跨线程使用?
  • leetcode:516 最长回文字序列 动态规划
  • 夫妻离婚对债务的约定是否对第三人有效?
  • [Android] [SnapdragonCamera] 单摄(横屏)阶段总结
  • GitHub每日最火火火项目(9.6)
  • Git 撤回commit
  • 主流的3D模型格式有什么区别?
  • <数据集>二维码识别数据集<目标检测>
  • 罗盘时钟屏保你见过吗?非常有特色的电脑时钟屏保
  • docker原理以及使用