Mutex,中文译为互斥体,在.net中也是作为一种线程或进程之间的互斥体存在。即在同一时刻,一个共享资源只允许被某一个线程或进程访问,其他线程或进程需要等待(直至获取互斥锁为止)。

Mutex的使用方式与Monitor很相似,但绝不相同。Monitor支持线程间并发同步,Mutex不但支持线程也支持进程间并发同步。

Mutex有许多需要注意的地方,如果稍不注意,则你会被坑惨!

接下来我们使用控制台和webapi项目各自做测试。

一、使用控制台测试进程间的同步

测试代码如下:

        [STAThread]
static void Main(string[] args)
{
//使用指定名称为MName的互斥体
//如果获取到名称为MName的互斥体,则flag为true,否则为false。flag作用很大,如果是指定了互斥体的名称,一定要用flag去判断是否互斥。
Mutex mutex = new Mutex(true, "MName", out bool flag); if (flag) {
Console.WriteLine("1.获取互斥锁,开始运行");
Thread.Sleep();
Console.WriteLine("2.释放互斥锁!");
mutex.ReleaseMutex();
Console.WriteLine("3.开始等待,5秒后超时!");
if (mutex.WaitOne())
Console.WriteLine("6.获取互斥锁,开始运行!");
else Console.WriteLine("6.超时释放互斥锁!");
Thread.Sleep();
}
else {
Console.WriteLine("1.未获取互斥锁,开始等待!");
mutex.WaitOne();
Console.WriteLine("4.获取互斥锁,开始运行!");
Console.WriteLine("5.释放互斥锁!");
mutex.ReleaseMutex();
} Console.ReadKey();
}

同时打开两个程序的运行结果如下所示:

进程间的并发同步需要指定互斥体的名称,只要是同一个名称,它们就能互斥。

Mutex的WaitOne()方法是阻塞当前线程并等待获取互斥体。ReleaseMutex()方法是释放当前线程拥有的互斥体,如果当前进程或线程没有拥有互斥体Mutex,则会抛出System.ApplicationException异常,这点需要注意。

WaitOne和ReleaseMutex方法最好成对出现,反正单个方法两两不相见。

多个ReleaseMutex方法一起运行的异常见以下图片:

这个问题可以看ReleaseMutex方法的定义说明:

    /// <summary>Releases the <see cref="T:System.Threading.Mutex"></see> once.</summary>
/// <exception cref="T:System.ApplicationException">The calling thread does not own the mutex.</exception>
/// <exception cref="T:System.ObjectDisposedException">The current instance has already been disposed.</exception>
public void ReleaseMutex();

产生System.ApplicationException异常的原因就是当前被调用的线程已经不拥有该互斥体!

二、使用WebApi项目测试线程间的同步

测试代码如下:

        [HttpGet("[controller]/v1/api/[action]")]
public IActionResult Test() {
return Json(SynchronizationTest());
} //指定一个互斥体名称
protected static string MutexName => "MName";
public ResponseModel SynchronizationTest() {
ResponseModel rc = new ResponseModel(, "初始化"); Mutex mutex = new Mutex(true, MutexName, out bool flag);
try {
//如果没有获取到互斥体,则等待,3秒后超时。如果获取到了互斥体,则flag=true,否则flag=false。一定要做flag判断,否则会抛出异常
if(!flag)
mutex.WaitOne(); int count = RedisHelper.Get(GoodsNumberKey).ToInt32();
if (count > ) {
RedisHelper.Set(GoodsNumberKey, "-1");
rc.SetMessage("重置成功!");
}
else rc.SetMessage("已被重置,本次重置无效"); rc.SetCode(Thread.CurrentThread.ManagedThreadId);
}
catch (Exception ex) {
_log.Error(ex);
}
finally {
mutex.ReleaseMutex();
mutex.Close();
mutex.Dispose();
} return rc;
}

接下来,我们使用如下代码连续发送8次请求

            HttpHelper.GetOnStringAsync($"{Url}/Shop/v1/api/Test", contentType: "application/json").ContinueWith(it =>
{
Console.WriteLine(it.Result);
});

测试的结果如下图所示:

可以看见,我们连续发送8次请求,只有一次重置成功,其他的都因为互斥的关系而重置无效。

.net core WebApi Mutex实现并发同步的更多相关文章

  1. .net core WebApi ManualResetEvent实现并发同步

    ManualResetEvent,即手动重置事件,通过信号量来判别当前线程是否应该阻塞或继续执行.使用方式与ManualResetEventSlim差不多,ManualResetEventSlim只是 ...

  2. .net core WebApi Monitor实现并发同步

    在.net中,还可以使用Monitor实现线程并发同步.Monitor类是纯托管且完全可移植,并且可能会在操作系统资源需求方面更加高效. Monitor的锁对象尽可能使用引用对象,如果是字符串或值对象 ...

  3. .Net Core WebAPI 基于Task的同步&异步编程快速入门

    .Net Core WebAPI 基于Task的同步&异步编程快速入门 Task.Result async & await 总结 并行任务(Task)以及基于Task的异步编程(asy ...

  4. Net Core WebAPI

    Net Core WebAPI .Net Core WebAPI 基于Task的同步&异步编程快速入门 Task.Result async & await 总结 并行任务(Task)以 ...

  5. .net core WebApi Interlocked配合ManualResetEventSlim实现并发同步

    由于项目有某种需求,在WebApi中,有大量的请求需要操作相同的数据,因此需要用到并发同步机制去操作共享的数据. 本次配合使用Interlocked和ManualResetEventSlim来实现并发 ...

  6. net core WebApi——文件分片下载

    目录 前言 开始 测试 小结 @ 前言 上一篇net core WebApi--文件分片上传与跨域请求处理介绍完文件的上传操作,本来是打算紧接着写文件下载,中间让形形色色的事给耽误的,今天还是抽个空整 ...

  7. ASP.NET Core WebAPI控制器返回类型的最佳选项

    前言 从.NET Core 2.1版开始,到目前为止,控制器操作可以返回三种类型的WebApi响应.这三种类型都有自己的优点和缺点,但都缺乏满足REST和高可测性的选项. ASP.NET Core中可 ...

  8. ASP.NET Core WebApi使用Swagger生成api说明文档

    1. Swagger是什么? Swagger 是一个规范和完整的框架,用于生成.描述.调用和可视化 RESTful 风格的 Web 服务.总体目标是使客户端和文件系统作为服务器以同样的速度来更新.文件 ...

  9. 使用 xUnit 编写 ASP.NET Core WebAPI单元测试

    本文使用xUnit对ASP.NET Core WebAPI做单元测试,使用HttpClient的同步和异步请求,下面详细介绍xUnit的使用过程: 一.创建示例项目 模板为我们自动创建了一个Value ...

随机推荐

  1. python将字符串转换成整型

    将字符串转换成,整型,从字面理解很容易让人误会. 比如,要把这个"abcabc"转换成整型,臣妾做不到啊.除成转成ascii. 我们所说字符串转成整型是这样的. s = " ...

  2. fiddler学习总结--利用fiddler快速模拟mock

    Mock的应用就是在测试接口的时候,去模拟我们想要的接口 1.创建一个txt文件,里面随意写一个json字符串,如图所示: 2.选择目标消息,并且点击“autoresponde”-->“add ...

  3. 第九篇——Struts2的拦截器

    拦截器: Struts2大多数核心功能都是通过拦截器实现的,每个拦截器完成某项功能: 拦截器方法在Action执行之前或之后执行. 工作原理: 拦截器的执行过程是一个递归的过程 action请求--& ...

  4. 收银台数据库存储AES加解密

    高级加密标准(AES,Advanced Encryption Standard)为最常见的对称加密算法加密和解密用到的密钥是相同的,这种加密方式加密速度非常快,适合经常发送数据的场合.缺点是密钥的传输 ...

  5. Docker Registry V2 with Nginx

    安装 nginx 修改/etc/yum.repos.d/nginx.repo [nginx] name=nginx repo baseurl=http://nginx.org/packages/cen ...

  6. python 装饰器(语法糖)

    def  login(func):    def testlogin():        for  i in range(3):            _username="abc" ...

  7. nginx 番外----添加第三方模块

    #第三方模块需要先进行下载,然后再编译时指定文件目录 1.查看当前编译模块 root@nginx sbin]# ./nginx -V #查看当前添加模块 nginx version: nginx/ b ...

  8. elasticsearch 索引备份恢复

    备份脚本 es_backup.sh : #!/bin/bash#备份昨天数据,删除30天前索引 host=`hostname`address="xxx@xxx.com" es_us ...

  9. [linux] grep 文本搜索工具

    grep [option] pattern file Linux系统中grep命令是一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来.grep全称是Global Regular ...

  10. 比原链Bytom错误码一览

    0XX API错误 BTM000", "Bytom API Error" 非比原标准错误 BTM001", "Request timed out&qu ...