.net core WebApi Mutex实现并发同步
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实现并发同步的更多相关文章
- .net core WebApi ManualResetEvent实现并发同步
ManualResetEvent,即手动重置事件,通过信号量来判别当前线程是否应该阻塞或继续执行.使用方式与ManualResetEventSlim差不多,ManualResetEventSlim只是 ...
- .net core WebApi Monitor实现并发同步
在.net中,还可以使用Monitor实现线程并发同步.Monitor类是纯托管且完全可移植,并且可能会在操作系统资源需求方面更加高效. Monitor的锁对象尽可能使用引用对象,如果是字符串或值对象 ...
- .Net Core WebAPI 基于Task的同步&异步编程快速入门
.Net Core WebAPI 基于Task的同步&异步编程快速入门 Task.Result async & await 总结 并行任务(Task)以及基于Task的异步编程(asy ...
- Net Core WebAPI
Net Core WebAPI .Net Core WebAPI 基于Task的同步&异步编程快速入门 Task.Result async & await 总结 并行任务(Task)以 ...
- .net core WebApi Interlocked配合ManualResetEventSlim实现并发同步
由于项目有某种需求,在WebApi中,有大量的请求需要操作相同的数据,因此需要用到并发同步机制去操作共享的数据. 本次配合使用Interlocked和ManualResetEventSlim来实现并发 ...
- net core WebApi——文件分片下载
目录 前言 开始 测试 小结 @ 前言 上一篇net core WebApi--文件分片上传与跨域请求处理介绍完文件的上传操作,本来是打算紧接着写文件下载,中间让形形色色的事给耽误的,今天还是抽个空整 ...
- ASP.NET Core WebAPI控制器返回类型的最佳选项
前言 从.NET Core 2.1版开始,到目前为止,控制器操作可以返回三种类型的WebApi响应.这三种类型都有自己的优点和缺点,但都缺乏满足REST和高可测性的选项. ASP.NET Core中可 ...
- ASP.NET Core WebApi使用Swagger生成api说明文档
1. Swagger是什么? Swagger 是一个规范和完整的框架,用于生成.描述.调用和可视化 RESTful 风格的 Web 服务.总体目标是使客户端和文件系统作为服务器以同样的速度来更新.文件 ...
- 使用 xUnit 编写 ASP.NET Core WebAPI单元测试
本文使用xUnit对ASP.NET Core WebAPI做单元测试,使用HttpClient的同步和异步请求,下面详细介绍xUnit的使用过程: 一.创建示例项目 模板为我们自动创建了一个Value ...
随机推荐
- 颜色扩展类--ColorExtensions
/// <summary> /// 颜色扩展类 /// </summary> public static class ColorExtensions { /// <sum ...
- kubernetes1.4新特性(一):支持sysctl命令
sysctl是一个允许改变正在运行中的Linux系统内核参数的接口.可以通过sysctl修改Linux系统内核中的TCP/IP 堆栈和虚拟内存系统的高级选项,而且不需要重新启动Linux系统,就可以实 ...
- Shell 文本处理命令
命令:cut –d’:’ -f1, 文件名 #切割处文件列的参数. -d切割字符. -f列的第几个参数. -c1-10指定字符串范围行的第一个到第十个. 命令:sort 文件名 #根据第一列第一个字符 ...
- 时间序列数据库调研之InfluxDB
基于 Go 语言开发,社区非常活跃,项目更新速度很快,日新月异,关注度高 测试版本 1.0.0_beta2-1 安装部署 wget https://dl.influxdata.com/influxdb ...
- weex安装失败,按照官网步骤多次失败后成功
在安装Weex Toolkit之前,需要确保安装了node, npm. yangfeifei:~ yff$ node -v v6.10.2 yangfeifei:~ yff$ npm -v 3.10. ...
- vue_elementUI_ tree树形控件 获取选中的父节点ID
el-tree 的 this.$refs.tree.getCheckedKeys() 只可以获取选中的id 无法获取选中的父节点ID想要获取选中父节点的id;需要如下操作1. 找到工程下的node_m ...
- [C++ Primer Plus] 第5章、循环和关系表达式(二)课后习题
1.编写一个要求用户输入两个整数的程序,将程序将计算并输出这两个整数之间(包括这两个整数)所有的整数的和.这里假设先输入较小的整数,例如如果用户输入的是2和9,则程序将指出2-9之间所有整数的和为44 ...
- JS设计模式(2)策略模式
什么是策略模式? 定义:根据不同参数可以命中不同的策略 主要解决:在有多种算法相似的情况下,使用 if...else 所带来的复杂和难以维护. 何时使用:有许多种情况,而区分它们的只是他们直接的行为. ...
- Convolutional Pose Machines
Convolutional Pose Machines 2018-12-10 18:17:20 Paper:https://www.cv-foundation.org/openaccess/conte ...
- 【Python】【自动化测试】【pytest】
https://docs.pytest.org/en/latest/getting-started.html#create-your-first-test http://www.testclass.n ...