背景:用户领取优惠券,同一个用户需要加锁验证是否已经领取,不同用户则可以同时领取。

上代码示例:

1、创建Person类

    /// <summary>
/// Person类
/// </summary>
public class Person
{
/// <summary>
/// id
/// </summary>
public int Id { get; set; } /// <summary>
/// 姓名
/// </summary>
public string Name { get; set; } /// <summary>
/// 是否获得优惠券
/// </summary>
public bool IsGetCoupon { get; set; }
}

2.1、不加锁的方法(可能会出现重复领取的情况)

        /// <summary>
/// 获取优惠券
/// </summary>
public static void GetCoupon(Person person)
{
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},前来领取优惠券", DateTime.Now, person.Name);
if (person.IsGetCoupon)
{
//假装业务处理
Thread.Sleep();
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},已经领取,不可重复领取", DateTime.Now, person.Name);
}
else
{
//假装业务处理
Thread.Sleep();
//领取
person.IsGetCoupon = true;
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},领取成功", DateTime.Now, person.Name);
}
}

2.2、加lock锁的方法,所有来领优惠券的人,都得排对领(也不好)

        /// <summary>
/// Lock获取优惠券
/// </summary>
public static void LockGetCoupon(Person person)
{
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},前来领取优惠券", DateTime.Now, person.Name);
lock (LockObj)
{
//判断是否已经领取
if (person.IsGetCoupon)
{
//假装业务处理
Thread.Sleep();
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},已经领取,不可重复领取", DateTime.Now, person.Name);
}
else
{
//假装业务处理
Thread.Sleep();
//领取
person.IsGetCoupon = true;
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},领取成功", DateTime.Now, person.Name);
}
}
}

2.3、mutex锁,互斥锁,只有相同id的人,才会排对领取,不同id的人就可以同时领取

        /// <summary>
/// Mutex,领取
/// </summary>
/// <param name="person"></param>
public static void MutexGetCoupon(Person person)
{
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},前来领取优惠券", DateTime.Now, person.Name);
using (var mutex = new Mutex(false, person.Id.ToString()))
{
try
{
if (mutex.WaitOne(-, false))
{
//判断是否已经领取
if (person.IsGetCoupon)
{
//假装业务处理
Thread.Sleep();
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},已经领取,不可重复领取", DateTime.Now, person.Name);
}
else
{ //假装业务处理
Thread.Sleep();
//领取
person.IsGetCoupon = true;
Console.WriteLine("date:{0:yyyy-MM-dd HH:mm:ss},name:{1},领取成功", DateTime.Now, person.Name);
}
}
}
catch (Exception ex)
{
//TxtLogHelper.WriteLog(ex);
}
finally
{
mutex.ReleaseMutex();
}
}
}
}

3.1、开始测试(不加锁)

        static void Main(string[] args)
{
//实例化三个人
Person p1 = new Person { Id = , Name = "Kobe" };
Person p2 = new Person { Id = , Name = "Rose" };
Person p3 = new Person { Id = , Name = "Lebl" }; //开启多线程、模拟三个人同时发起多次领取请求
for (int i = ; i < ; i++)
{
new Thread(() =>
{
GetCoupon(p1);
}).Start();
new Thread(() =>
{
GetCoupon(p2);
}).Start();
new Thread(() =>
{
GetCoupon(p3);
}).Start();
}
Console.ReadLine();
}

测试结果:每个人都重复领取

3.2、测试lock锁方法,

        private static readonly object LockObj = new object();
static void Main(string[] args)
{
//实例化三个人
Person p1 = new Person { Id = , Name = "Kobe" };
Person p2 = new Person { Id = , Name = "Rose" };
Person p3 = new Person { Id = , Name = "Lebl" }; //开启多线程、模拟三个人同时发起多次领取请求
for (int i = ; i < ; i++)
{
new Thread(() =>
{
LockGetCoupon(p1);
}).Start();
new Thread(() =>
{
LockGetCoupon(p2);
}).Start();
new Thread(() =>
{
LockGetCoupon(p3);
}).Start();
}
Console.ReadLine();
}

测试结果:虽然避免了重复领取,但是每个人都的每个请求都要排对。如果用户量大的话,这种方式效率就太低了,所以不推荐。

3.3、测试mutex锁,互斥锁

        static void Main(string[] args)
{
//实例化三个人
Person p1 = new Person { Id = , Name = "Kobe" };
Person p2 = new Person { Id = , Name = "Rose" };
Person p3 = new Person { Id = , Name = "Lebl" }; //开启多线程、模拟三个人同时发起多次领取请求
for (int i = ; i < ; i++)
{
new Thread(() =>
{
MutexGetCoupon(p1);
}).Start();
new Thread(() =>
{
MutexGetCoupon(p2);
}).Start();
new Thread(() =>
{
MutexGetCoupon(p3);
}).Start();
}
Console.ReadLine();
}

测试结果:既避免了重复领取,也避免了堵塞用户请求的情况。见下面截图,Kobe、Rose、Lebl是同时领取的优惠券,但是每个人的重复请求都在排对

总结:mutex锁,完美的解决了此类问题。

--------------------------------------------华丽的分割线 --------------------------------------------

感谢各位大佬提出的问题和建议,我确实没有考虑到这些问题。

C# 针对特定的条件进行锁操作,不用lock,而是mutex的更多相关文章

  1. Microsoft Dynamics CRM 2013 --针对特定实体,取消保存功能(包含自动保存)

    AutoSave 是 Microsoft Dynamics CRM 2013 一个新特性. 但AutoSave的有效范围是[所有实体],不允许针对特定某一实体进行设置. 所以,若想针对特定实体进行设置 ...

  2. 针对特定XML的解析器XMLParser

    一.建立网页库和偏移文件 为文本搜索引擎建立网页库,首先要把所有的网页(这里是文章)格式化,并保存到指定的格式中.如以下格式:   |                                  ...

  3. MySQL 事务的隔离级别及锁操作的一点点演示

    MySQL 版本:5.7 安装环境:MAC OS 一.测试数据 测试数据库:test:测试表:tt CREATE TABLE `tt` ( `id` int(11) DEFAULT NULL, `na ...

  4. html小知识点汇总(浏览器导航上显示图标、div无高度时试着清除浮动、文字环绕图片、字体加粗、div按百分比分、已有的不合适的class,针对特定的标签进行修改)

    1.新点击的网页,在浏览器导航上显示图标: 像这种效果: <head> <meta charset="UTF-8"> <meta name=" ...

  5. 更新xcode后插件失效问题——不针对特定版本的通用解决方法

    一.Xcode更新后插件失效的原理 1.每次更新Xcode后插件都会失效,其实插件都还在这个目录好好的躺着呢: ~/Library/Application Support/Developer/Shar ...

  6. 链表的无锁操作 (JAVA)

    看了下网上关于链表的无锁操作,写的不清楚,遂自己整理一部分,主要使用concurrent并发包的CAS操作. 1. 链表尾部插入 待插入的节点为:cur 尾节点:pred 基本插入方法: do{ pr ...

  7. Mysql优化(出自官方文档) - 第十二篇(优化锁操作篇)

    Mysql优化(出自官方文档) - 第十二篇(优化锁操作篇) 目录 Mysql优化(出自官方文档) - 第十二篇(优化锁操作篇) 1 Internal Locking Methods Row-Leve ...

  8. PHP中针对区域语言标记信息的操作

    相信大家对 zh_CN 这个东西绝对不会陌生,不管是 PHP 中,还是在我们的网页上,都会见到它的身影.其实这就是指定我们的显示编码是什么国家或者地区的,使用何种语言.对于这种区域语言的标记来说,PH ...

  9. **解释器全局锁(Global Interpreter Lock)

    解释器全局锁(Global Interpreter Lock),即Python为了保证线程安全而采取的独立线程运行的限制,说白了就是一个核只能在同一时间运行一个线程. [解决办法就是多进程和协程(协程 ...

随机推荐

  1. Morris 轻量级 图表

    Morris.js 是基于 jQuery 和 Raphaël 的轻量级矢量图形库,帮助开发人员轻松绘制各种形式的图表.示例: HTML: <div id="myfirstchart&q ...

  2. Elevated privileges for Delphi applications

    BY CRAIG CHAPMAN · PUBLISHED 2015-06-08 · UPDATED 2015-06-08   One of my customers recently asked th ...

  3. [Erlang-0016][aque_tcp] 一个 Erlang TCP 组件

    项目地址:https://github.com/liangjingyang/aque_tcp 欢迎任何形式的转载,但请务必注明出处:http://www.cnblogs.com/liangjingya ...

  4. 常见的几个Qt编程问题的处理(转自QT中文论坛)(挺实用的)

    1.如何在窗体关闭前自行判断是否可关闭答:重新实现这个窗体的closeEvent()函数,加入判断操作 void MainWindow::closeEvent(QCloseEvent*event){i ...

  5. Qt之界面数据存储与获取(userData)

    http://blog.csdn.net/u011012932/article/details/52413012#comments

  6. C#每天进步一点--引用类型和值类型

    在刚参加工作面试时,我们经常会遇到有关值类型和引用类型的问题,你回答的怎么样直接影响你在别人心目中的印象,你回答的不好说明你对C#没有深入的了解学习,今天我带大家回顾下C#中的引用类型和值类型. CL ...

  7. hgoi#20190514

    T1-Curriculum Vitae 给你一个长度为n的01序列a,删去其中的几个数,使得序列中左边是连续的0,右边是连续的1,可以没有0或1,求最多剩下几个数 解法 对于每个点看它左边几个0,右边 ...

  8. Go语言学习——彻底弄懂return和defer的微妙关系

    疑问 前面在函数篇里介绍了Go语言的函数是支持多返回值的. 只要在函数体内,对返回值赋值,最后加上return就可以返回所有的返回值. 最近在写代码的时候经常遇到在return后,还要在defer里面 ...

  9. 记一次腾讯IEG面试失败经历

    如果这是一次成功的经历,估计浏览量不会低.无奈本人能力有限,而且一直在实习,准备时间与面试经验有限导致此次失败,不过,失败也是一种宝贵的经验,我希望也相信这里能给大家一些比较珍贵的经验,废话不多说,上 ...

  10. Angular4.0从入门到实战打造在线竞拍网站学习笔记之一--组件

    Angular4.0基础知识之组件 Angular4.0基础知识之路由 Angular4.0依赖注入 Angular4.0数据绑定&管道 最近搞到手了一部Angular4的视频教程,这几天正好 ...