看Elsa-core源代码中看到的,Elsa-core中所有保存数据的方法似乎使用同一个Save方法。如下图:

那么为什么要使用这玩意,我还是头一次见这玩意????

好吧,我承认我自己菜。我自个儿也该保持谦虚态度学习学习了。

先看下这个SemaphoreSlim类的描述

(我买的正版Resharp 2022)反编译代码如上图。

翻译过来就是 “限制当前访问资源或池中资源的线程数”,真的是这样的吗? 试一试...

我的代码仓库  https://github.com/qiqiqiyaya/Learning-Case/tree/main/SemaphoreSlim ,测试例子。

例一

在B乎中看到一篇文章 ,链接地址  https://zhuanlan.zhihu.com/p/158777952

我copy他的例子测试了下,如下图:

// 现在有10个人要过桥
// 但是一座桥上只能承受5个人,再多桥就会塌
static void SemaphoreTest()
{
var semaphore = new SemaphoreSlim(5);
for (int i = 1; i <= 10; i++)
{
Thread.Sleep(100); // 排队上桥
var index = i; // 定义index 避免出现闭包的问题
Task.Run(() =>
{
semaphore.Wait();
try
{
Console.WriteLine($"第{index}个人正在过桥。");
Thread.Sleep(5000); // 模拟过桥需要花费的时间
}
finally
{
Console.WriteLine($"第{index}个人已经过桥。");
semaphore.Release();
}
});
}
}

运行结果,与B乎上作者的结果一样。

这里有个问题,该作者使用Task.Run 然后在其中添加了 Thread.Sleep(5000); ,这会阻塞当前的线程,导致执行 Task 任务的“任务调度器”开启了 10个线程。

关于什么是 Task ,理解 Task ,请阅读大佬的文章,非常Nice链接 https://www.cnblogs.com/artech/p/task_scheduling.html

列二,如果在异步Async/Await中是什么情况呢?

static void SemaphoreTest1()
{
var semaphore = new SemaphoreSlim(5);
for (int i = 1; i <= 10; i++)
{
Thread.Sleep(100); // 排队上桥
var index = i; // 定义index 避免出现闭包的问题
Task.Run(async () =>
{
Console.WriteLine($"第{index}个人已抵达桥边上。线程Id " + Thread.CurrentThread.ManagedThreadId);
semaphore.Wait();
try
{
Console.WriteLine($"第{index}个人正在过桥。线程Id " + Thread.CurrentThread.ManagedThreadId);
//Thread.Sleep(5000); // 模拟过桥需要花费的时间
await Task.Delay(5000);
}
finally
{
Console.WriteLine($"第{index}个人已经过桥。线程Id " + Thread.CurrentThread.ManagedThreadId);
semaphore.Release();
}
});
}
}

将 Thread.Sleep(5000); 换成了 await Task.Delay(5000); 使用 async/await 。结果如下:

从结果上看,例如:第1个人 线程Id 3 上桥,在执行 await Task.Delay(5000); (模拟过桥需要花费的时间)后,线程Id变成了 12 。这是执行 Task 的任务调度器(Task默认调度器是线程池)作用的效果。这不是本次随笔的重点。

再次强调关于理解Task,请参考 https://www.cnblogs.com/artech/p/task_scheduling.html 。

再次强调Task与线程Thread是两个东西,Task可以理解为一个任务,那么这个任务由Thread去执行,至于由哪一个Thread 去执行,这就由 “任务调度器” 去决定了。

从结果上看,执行了五个Task之后,就阻止后续Task再往下执行代码了。

列二中使用了 semaphore.Wait(); 同步阻塞 , 导致后续Task中使用一个之前空闲的Id为3的线程 (Id为3的线程执行到 await Task.Delay(5000) 后阻塞了,该线程被调度器拿去执行新的Task任务),并创建了另外4个新的执行Task的线程。

结论:

1.SemaphoreSlim会限制访问资源的线程数

2.在异步async/await情况下,SemaphoreSlim会限制正在执行访问资源的Task的数量

例三

static void SemaphoreTest2()
{
var semaphore = new SemaphoreSlim(5);
for (int i = 1; i <= 10; i++)
{
Thread.Sleep(100); // 排队上桥
var index = i; // 定义index 避免出现闭包的问题
Task.Run(async () =>
{
Console.WriteLine($"第{index}个人已抵达桥边上。线程Id " + Thread.CurrentThread.ManagedThreadId);
await semaphore.WaitAsync();
try
{
Console.WriteLine($"第{index}个人正在过桥。线程Id " + Thread.CurrentThread.ManagedThreadId);
await Task.Delay(5000);// 模拟过桥需要花费的时间
}
finally
{
Console.WriteLine($"第{index}个人已经过桥。线程Id " + Thread.CurrentThread.ManagedThreadId);
semaphore.Release();
}
});
}
}

执行结果

例三中我使用了 await semaphore.WaitAsync(); ,也就是说不存在同步阻塞代码了,全部都是异步。

异步下遇到await,执行等待,那么当前执行这个Task的线程可能会被“任务调度器拿去执行别的Task”

结论:

1.在异步async/await情况下,SemaphoreSlim会限制正在执行访问资源的Task的数量,  与例二第二个结论一致。

异步下的Lock锁

鉴于SemaphoreSlim的特殊性,可以使用其通过程序角度来实现“”(叫乐观锁、还是悲观锁???,我去,两者好像都不是

乐观锁与悲观锁定义如下图

如下图代码:

重点代码  SemaphoreSlim _semaphore = new(1) 

如果真要说明,我觉得是最多像 悲观锁 ,但又不全是。这里的SemaphoreSlim只是说同一时刻只准一个线程或正在执行的Task做保存操作,那么如果是分布式程序呢,另一个程序此时也执行保存操作?

我觉得其真实原因是:使用SemaphoreSlim可以实现异步下的 Lock 锁。

.Net SemaphoreSlim的更多相关文章

  1. 【C#】【Thread】Semaphore/SemaphoreSlim信号量

    System.Threading.Semaphore 类表示一个命名(系统范围)信号量或本地信号量. 它是一个对 Win32 信号量对象的精简包装. Win32 信号量是计数信号量,可用于控制对资源池 ...

  2. C#并行编程 (Barrier,CountdownEvent,ManualResetEventSlim,SemaphoreSlim,SpinLock,SpinWait )

    背景 有时候必须访问变量.实例.方法.属性或者结构体,而这些并没有准备好用于并发访问,或者有时候需要执行部分代码,而这些代码必须单独运行,这是不得不通过将任务分解的方式让它们独立运行. 当任务和线程要 ...

  3. 第十三节:实际开发中使用最多的监视锁Monitor、lock语法糖的扩展、混合锁的使用(ManualResetEvent、SemaphoreSlim、ReaderWriterLockSlim)

    一. 监视锁(Monitor和lock) 1. Monitor类,限定线程个数的一把锁,两个核心方法: Enter:锁住某个资源. Exit:退出某一个资源. 测试案例:开启5个线程同时对一个变量进行 ...

  4. C# SemaphoreSlim 实现

    当多个任务或线程并行运行时,难以避免的对某些有限的资源进行并发的访问.可以考虑使用信号量来进行这方面的控制(System.Threading.Semaphore)是表示一个Windows内核的信号量对 ...

  5. 《C#多线程编程实战》2.4 SemaphoreSlim

    这个简单多了. 理解也是很好理解. 比上一个mutex好理解多了. 这个SemaphoreSlim是干什么呢? 就是限制线程的来访问. 好比说一次只有两个,一次只有三个  这样的线程来访问资源. 有点 ...

  6. Semaphore and SemaphoreSlim

    https://msdn.microsoft.com/en-us/library/z6zx288a(v=vs.110).aspx The System.Threading.Semaphore clas ...

  7. SemaphoreSlim

    https://msdn.microsoft.com/en-us/library/system.threading.semaphoreslim(v=vs.110).aspx Represents a ...

  8. 一次 .NET Core 中玩锁的经历:ManualResetEventSlim, Semaphore 与 SemaphoreSlim

    最近同事对  .net core memcached 缓存客户端 EnyimMemcachedCore 进行了高并发下的压力测试,发现在 linux 上高并发下使用 async 异步方法读取缓存数据会 ...

  9. Mutex vs Semaphore vs Monitor vs SemaphoreSlim

    C#开发者(面试者)都会遇到Mutex,Semaphore,Monitor,SemaphoreSlim这四个与锁相关的C#类型,本文期望以最简洁明了的方式阐述四种对象的区别. 线程安全 教条式理解 如 ...

  10. SemaphoreSlim 实现

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/dz45693/article/deta ...

随机推荐

  1. KingbaseES R6 通过脚本构建集群案例

      案例说明: KingbaseES V8R6部署一般可采用图形化方式快速部署,但在生产一线,有的服务器系统未启用图形化环境,所以对于KingbaseES V8R6的集群需采用手工字符界面方式部署,本 ...

  2. 【ASP.NET Core】自定义Session的存储方式

    在开始今天的表演之前,老周先跟大伙伴们说一句:"中秋节快乐". 今天咱们来聊一下如何自己动手,实现会话(Session)的存储方式.默认是存放在分布式内存中.由于HTTP消息是无状 ...

  3. python一招完美搞定Chromedriver的自动更新

    日常的web自动化过程中,我们常常用python selenium库来操纵Chrome浏览器实现网页的自动化.这其中有个比较头疼的问题:Chrome的更新频率非常频繁,与之对应的Chromedrive ...

  4. G&GH04 本地连接至远程

    注意事项与声明 平台: Windows 10 作者: JamesNULLiu 邮箱: jamesnulliu@outlook.com 博客: https://www.cnblogs.com/james ...

  5. 在 Kubernetes 容器集群,微服务项目最佳实践

    转载自:https://mp.weixin.qq.com/s/WYu3gDwKKf06f_FYbO9YRg 本文主要介绍我个人在使用 Kubernetes 的过程中,总结出的一套「Kubernetes ...

  6. 【ceph】理解Ceph的三种存储接口:块设备、文件系统、对象存储

    文章转载自:https://blog.51cto.com/liangchaoxi/4049104

  7. 第二章:视图层 - 9:动态生成CSV文件

    CSV (Comma Separated Values),以纯文本形式存储数字和文本数据的存储方式.纯文本意味着该文件是一个字符序列,不含必须像二进制数字那样的数据.CSV文件由任意数目的记录组成,记 ...

  8. 使用二进制文件部署Etcd集群

    Etcd 是一个分布式键值存储系统,Kubernetes使用Etcd进行数据存储,所以先准备一个Etcd数据库,为解决Etcd单点故障,应采用集群方式部署,这里使用3台组建集群,可容忍1台机器故障,当 ...

  9. 编码中的Adapter,不仅是一种设计模式,更是一种架构理念与解决方案

    大家好,又见面了. 不知道下面这玩意大家有没有见过或者使用过?这是一个插座转换器.我们都知道日常使用的是220v的交流电,而国外不同国家使用的电流电压是不一样的(比如日本使用的是110v).且插座的接 ...

  10. Linux Subsystem For Android 11!适用于Debian GNU/Linux的Android子系统,完美兼容ARM安卓软件!

    本文将讲述如何在Debian Stable 系统安装一个Android 11子系统,并且这个子系统带有Houdini可以兼容专为移动设备开发的ARM软件.在root权限下,编辑/etc/apt/sou ...