第二十节: 深入理解并发机制以及解决方案(锁机制、EF自有机制、队列模式等)
一. 理解并发机制
1. 什么是并发,并发与多线程有什么关系?
①. 先从广义上来说,或者从实际场景上来说.
高并发通常是海量用户同时访问(比如:12306买票、淘宝的双十一抢购),如果把一个用户看做一个线程的话那么并发可以理解成多线程同时访问,高并发即海量线程同时访问。
(ps:我们在这里模拟高并发可以for循环多个线程即可)
②.从代码或数据的层次上来说.
多个线程同时在一条相同的数据上执行多个数据库操作。
2. 从代码层次上来说,给并发分类。
①.积极并发(乐观并发、乐观锁):无论何时从数据库请求数据,数据都会被读取并保存到应用内存中。数据库级别没有放置任何显式锁。数据操作会按照数据层接收到的先后顺序来执行。
积极并发本质就是允许冲突发生,然后在代码本身采取一种合理的方式去解决这个并发冲突,常见的方式有:
a.忽略冲突强制更新:数据库会保存最后一次更新操作(以更新为例),会损失很多用户的更新操作。
b.部分更新:允许所有的更改,但是不允许更新完整的行,只有特定用户拥有的列更新了。这就意味着,如果两个用户更新相同的记录但却不同的列,那么这两个更新都会成功,而且来自这两个用户的更改都是可见的。(EF默认实现不了这种情况)
c.询问用户:当一个用户尝试更新一个记录时,但是该记录自从他读取之后已经被别人修改了,这时应用程序就会警告该用户该数据已经被某人更改了,然后询问他是否仍然要重写该数据还是首先检查已经更新的数据。(EF可以实现这种情况,在后面详细介绍)
d.拒绝修改:当一个用户尝试更新一个记录时,但是该记录自从他读取之后已经被别人修改了,此时告诉该用户不允许更新该数据,因为数据已经被某人更新了。
(EF可以实现这种情况,在后面详细介绍)
②.消极并发(悲观并发、悲观锁):无论何时从数据库请求数据,数据都会被读取,然后该数据上就会加锁,因此没有人能访问该数据。这会降低并发出现问题的机会,缺点是加锁是一个昂贵的操作,会降低整个应用程序的性能。
消极并发的本质就是永远不让冲突发生,通常的处理凡是是只读锁和更新锁。
a. 当把只读锁放到记录上时,应用程序只能读取该记录。如果应用程序要更新该记录,它必须获取到该记录上的更新锁。如果记录上加了只读锁,那么该记录仍然能够被想要只读锁的请求使用。然而,如果需要更新锁,该请求必须等到所有的只读锁释放。同样,如果记录上加了更新锁,那么其他的请求不能再在这个记录上加锁,该请求必须等到已存在的更新锁释放才能加锁。
总结,这里我们可以简单理解把并发业务部分用一个锁(如:lock,实质是数据库锁,后面章节单独介绍)锁住,使其同时只允许一个线程访问即可。
b. 加锁会带来很多弊端:
(1):应用程序必须管理每个操作正在获取的所有锁;
(2):加锁机制的内存需求会降低应用性能
(3):多个请求互相等待需要的锁,会增加死锁的可能性。
总结:尽量不要使用消极并发,EF默认是不支持消极并发的!!!
注意:EF默认就是积极并发,当然EF也可以配置成消极并发。
二. 并发机制的解决方案
1. 从架构的角度去解决(大层次 如:12306买票)
nginx负载均衡、数据库读写分离、多个业务服务器、多个数据库服务器、NoSQL, 使用队列来处理业务,将高并发的业务依次放到队列中,然后按照先进先出的原则, 逐个处理(队列的处理可以采用 Redis、RabbitMq等等)
(PS:在后面的框架篇章里详细介绍该方案)
2. 从代码的角度去解决(在服务器能承载压力的情况下,并发访问同一条数据)
实际的业务场景:如进销存类的项目,涉及到同一个物品的出库、入库、库存,我们都知道库存在数据库里对应了一条记录,入库要查出现在库存的数量,然后加上入库的数量,假设两个线程同时入库,假设查询出来的库存数量相同,但是更新库存数量在数据库层次上是有先后,最终就保留了后更新的数据,显然是不正确的,应该保留的是两次入库的数量和。
(该案例的实质:多个线程同时在一条相同的数据上执行多个数据库操作)
事先准备一张数据库表:
解决方案一:(最常用的方式)
给入库和出库操作加一个锁,使其同时只允许一个线程访问,这样即使两个线程同时访问,但在代码层次上,由于锁的原因,还是有先有后的,这样就保证了入库操作的线程唯一性,当然库存量就不会出错了.
总结:该方案可以说是适合处理小范围的并发且锁内的业务执行不是很复杂。假设一万线程同时入库,每次入库要等2s,那么这一万个线程执行完成需要的总时间非常多,显然不适合。
(这种方式的实质就是给核心业务加了个lock锁,这里就不做测试了)
解决方案二:EF处理积极并发带来的冲突
1. 配置准备
(1). 针对DBFirst模式,可以给相应的表额外加一列RowVersion,数据库中为timestamp类型,对应的类中为byte[]类型,并且在Edmx模型上给该字段的并发模式设置为fixed(默认为None),这样该表中所有字段都监控并发。
如果不想监视所有列(在不添加RowVersion的情况下),只需在Edmx模型是给特定的字段的并发模式设置为fixed,这样只有被设置的字段被监测并发。
测试结果: (DBFirst模式下的并发测试)
事先在UserInfor1表中插入一条id、userName、userSex、userAge均为1的数据(清空数据)。
测试情况1:
在不设置RowVersion并发模式为Fixed的情况下,两个线程修改不同字段(修改同一个字段一个道理),后执行的线程的结果覆盖前面的线程结果.
发现测试结果为:1,1,男,1 ; 显然db1线程修改的结果被db2线程给覆盖了. (修改同一个字段一个道理)
{
//1.创建两个EF上下文,模拟代表两个线程
var db1 = new ConcurrentTestDBEntities();
var db2 = new ConcurrentTestDBEntities(); UserInfor1 user1 = db1.UserInfor1.Find("");
UserInfor1 user2 = db2.UserInfor1.Find(""); //2. 执行修改操作
//(db1的线程先执行完修改操作,并保存)
user1.userName = "ypf";
db1.Entry(user1).State = EntityState.Modified;
db1.SaveChanges(); //(db2的线程在db1线程修改完成后,执行修改操作)
try
{
user2.userSex = "男";
db2.Entry(user2).State = EntityState.Modified;
db2.SaveChanges(); Console.WriteLine("测试成功");
}
catch (Exception)
{
Console.WriteLine("测试失败");
}
}
测试情况2:
设置RowVersion并发模式为Fixed的情况下,两个线程修改不同字段(修改同一个字段一个道理),如果该条数据已经被修改,利用DbUpdateConcurrencyException可以捕获异常,进行积极并发的冲突处理。测试结果如下:
a.RefreshMode.ClientWins: 1,1,男,1
b.RefreshMode.StoreWins: 1,ypf,1,1
c.ex.Entries.Single().Reload(); 1,ypf,1,1
{
//1.创建两个EF上下文,模拟代表两个线程
var db1 = new ConcurrentTestDBEntities();
var db2 = new ConcurrentTestDBEntities(); UserInfor1 user1 = db1.UserInfor1.Find("");
UserInfor1 user2 = db2.UserInfor1.Find(""); //2. 执行修改操作
//(db1的线程先执行完修改操作,并保存)
user1.userName = "ypf";
db1.Entry(user1).State = EntityState.Modified;
db1.SaveChanges(); //(db2的线程在db1线程修改完成后,执行修改操作)
try
{
user2.userSex = "男";
db2.Entry(user2).State = EntityState.Modified;
db2.SaveChanges(); Console.WriteLine("测试成功");
}
catch (DbUpdateConcurrencyException ex)
{
Console.WriteLine("测试失败:" + ex.Message); //1. 保留上下文中的现有数据(即最新,最后一次输入)
//var oc = ((IObjectContextAdapter)db2).ObjectContext;
//oc.Refresh(RefreshMode.ClientWins, user2);
//oc.SaveChanges(); //2. 保留原始数据(即数据源中的数据代替当前上下文中的数据)
//var oc = ((IObjectContextAdapter)db2).ObjectContext;
//oc.Refresh(RefreshMode.StoreWins, user2);
//oc.SaveChanges(); //3. 保留原始数据(而Reload处理也就是StoreWins,意味着放弃当前内存中的实体,重新到数据库中加载当前实体)
ex.Entries.Single().Reload();
db2.SaveChanges();
}
}
测试情况3:
在不设置RowVersion并发模式为Fixed的情况下(也不需要RowVersion这个字段),单独设置userName字段的并发模式为Fixed,两个线程同时修改该字段,利用DbUpdateConcurrencyException可以捕获异常,进行积极并发的冲突处理,但如果是两个线程同时修改userName以外的字段,将不能捕获异常,将走EF默认的处理方式,后执行的覆盖先执行的。
a.RefreshMode.ClientWins: 1,ypf2,1,1
b.RefreshMode.StoreWins: 1,ypf,1,1
c.ex.Entries.Single().Reload(); 1,ypf,1,1
{
//1.创建两个EF上下文,模拟代表两个线程
var db1 = new ConcurrentTestDBEntities();
var db2 = new ConcurrentTestDBEntities(); UserInfor1 user1 = db1.UserInfor1.Find("");
UserInfor1 user2 = db2.UserInfor1.Find(""); //2. 执行修改操作
//(db1的线程先执行完修改操作,并保存)
user1.userName = "ypf";
db1.Entry(user1).State = EntityState.Modified;
db1.SaveChanges(); //(db2的线程在db1线程修改完成后,执行修改操作)
try
{
user2.userName = "ypf2";
db2.Entry(user2).State = EntityState.Modified;
db2.SaveChanges(); Console.WriteLine("测试成功");
}
catch (DbUpdateConcurrencyException ex)
{
Console.WriteLine("测试失败:" + ex.Message); //1. 保留上下文中的现有数据(即最新,最后一次输入)
var oc = ((IObjectContextAdapter)db2).ObjectContext;
oc.Refresh(RefreshMode.ClientWins, user2);
oc.SaveChanges(); //2. 保留原始数据(即数据源中的数据代替当前上下文中的数据)
//var oc = ((IObjectContextAdapter)db2).ObjectContext;
//oc.Refresh(RefreshMode.StoreWins, user2);
//oc.SaveChanges(); //3. 保留原始数据(而Reload处理也就是StoreWins,意味着放弃当前内存中的实体,重新到数据库中加载当前实体)
//ex.Entries.Single().Reload();
//db2.SaveChanges();
}
}
(2). 针对CodeFirst模式,需要有这样的一个属性 public byte[] RowVersion { get; set; },并且给属性加上特性[Timestamp],这样该表中所有字段都监控并发。如果不想监视所有列(在不添加RowVersion的情况下),只需给特定的字段加上特性 [ConcurrencyCheck],这样只有被设置的字段被监测并发。
除了再配置上不同于DBFirst模式以为,是通过加特性的方式来标记并发,其它捕获并发和积极并发的几类处理方式均同DBFirst模式相同。(这里不做测试了)
2. 积极并发处理的三种形式总结:
利用DbUpdateConcurrencyException可以捕获异常,然后:
a. RefreshMode.ClientWins:保留上下文中的现有数据(即最新,最后一次输入)
b. RefreshMode.StoreWins:保留原始数据(即数据源中的数据代替当前上下文中的数据)
c.ex.Entries.Single().Reload(); 保留原始数据(而Reload处理也就是StoreWins,意味着放弃当前内存中的实体,重新到数据库中加载当前实体)
3. 该方案总结:
这种模式实质上就是获取异常告诉程序,让开发人员结合需求自己选择怎么处理,但这种模式是解决代码层次上的并发冲突,并不是解决大数量同时访问崩溃问题的。
解决方案三:利用队列来解决业务上的并发(架构层次上其实也是这种思路解决的)
1.先分析:
前面说过所谓的高并发,就是海量的用户同时向服务器发送请求,进行某个业务处理(比如定时秒杀的抢单),而这个业务处理是需要 一定时间的。
2.处理思路:
将海量用户的请求放到一个队列里(如:Queue),先不进行业务处理,然后另外一个服务器从线程中读取这个请求(MVC框架可以放到Global全局里),依次进行业务处理,至于处理完成后,是否需要告诉客户端,可以根据实际需求来定,如果需要的话(可以借助Socket、Signalr、推送等技术来进行).
特别注意:读取队列的线程是一直在运行,只要队列中有数据,就给他拿出来.
这里使用Queue队列,可以参考:http://www.cnblogs.com/yaopengfei/p/8322016.html
(PS:架构层次上的处理方案无非队列是单独一台服务器,执行从队列读取的是另外一台业务服务器,处理思想是相同的)
队列单例类的代码:
/// <summary>
/// 单例类
/// </summary>
public class QueueUtils
{
/// <summary>
/// 静态变量:由CLR保证,在程序第一次使用该类之前被调用,而且只调用一次
/// </summary>
private static readonly QueueUtils _QueueUtils = new QueueUtils(); /// <summary>
/// 声明为private类型的构造函数,禁止外部实例化
/// </summary>
private QueueUtils()
{ }
/// <summary>
/// 声明属性,供外部调用,此处也可以声明成方法
/// </summary>
public static QueueUtils instanse
{
get
{
return _QueueUtils;
}
} //下面是队列相关的
System.Collections.Queue queue = new System.Collections.Queue(); private static object o = new object(); public int getCount()
{
return queue.Count;
} /// <summary>
/// 入队方法
/// </summary>
/// <param name="myObject"></param>
public void Enqueue(object myObject)
{
lock (o)
{
queue.Enqueue(myObject);
}
}
/// <summary>
/// 出队操作
/// </summary>
/// <returns></returns>
public object Dequeue()
{
lock (o)
{
if (queue.Count > )
{
return queue.Dequeue();
}
}
return null;
} }
PS:这里的入队和出队都要加锁,因为Queue默认不是线程安全的,不加锁会存在资源竞用问题从而业务出错,或者直接使用ConcurrentQueue线程安全的队列,就不需要加锁了,关于队列线程安全问题详见:http://www.cnblogs.com/yaopengfei/p/8322016.html
临时存储数据类的代码:
/// <summary>
/// 该类用来存储请求信息
/// </summary>
public class TempInfor
{
/// <summary>
/// 用户编号
/// </summary>
public string userId { get; set; }
}
模拟高并发入队,单独线程出队的代码:
{
//3.1 模拟高并发请求 写入队列
{
for (int i = ; i < ; i++)
{
Task.Run(() =>
{
TempInfor tempInfor = new TempInfor();
tempInfor.userId = Guid.NewGuid().ToString("N");
//下面进行入队操作
QueueUtils.instanse.Enqueue(tempInfor); });
}
}
//3.2 模拟另外一个线程队列中读取数据请求标记,进行相应的业务处理(该线程一直运行,不停止)
Task.Run(() =>
{
while (true)
{
if (QueueUtils.instanse.getCount() > )
{
//下面进行出队操作
TempInfor tempInfor2 = (TempInfor)QueueUtils.instanse.Dequeue(); //拿到请求标记,进行相应的业务处理
Console.WriteLine("id={0}的业务执行成功", tempInfor2.userId);
}
}
});
//3.3 模拟过了一段时间(6s后),又有新的请求写入
Thread.Sleep();
Console.WriteLine("6s的时间已经过去了");
{
for (int j = ; j < ; j++)
{
Task.Run(() =>
{
TempInfor tempInfor = new TempInfor();
tempInfor.userId = Guid.NewGuid().ToString("N");
//下面进行入队操作
QueueUtils.instanse.Enqueue(tempInfor); });
}
}
}
3.下面案例的测试结果:
一次输出100条数据,6s过后,再一次输出100条数据。
4. 总结:
该方案是一种迂回的方式处理高并发,在业内这种思想也是非常常见,但该方案也有一个弊端,客户端请求的实时性很难保证,或者即使要保证(比如引入实时通讯技术),
也要付出不少代价.
解决方案四: 利用数据库自有的锁机制进行处理
(在后面数据锁机制章节进行介绍)
!
- 作 者 : Yaopengfei(姚鹏飞)
- 博客地址 : http://www.cnblogs.com/yaopengfei/
- 声 明1 : 本人才疏学浅,用郭德纲的话说“我是一个小学生”,如有错误,欢迎讨论,请勿谩骂^_^。
- 声 明2 : 原创博客请在转载时保留原文链接或在文章开头加上本人博客地址,否则保留追究法律责任的权利。
第二十节: 深入理解并发机制以及解决方案(锁机制、EF自有机制、队列模式等)的更多相关文章
- centos LAMP第二部分apache配置 下载discuz!配置第一个虚拟主机 安装Discuz! 用户认证 配置域名跳转 配置apache的访问日志 配置静态文件缓存 配置防盗链 访问控制 apache rewrite 配置开机启动apache tcpdump 第二十节课
centos LAMP第二部分apache配置 下载discuz!配置第一个虚拟主机 安装Discuz! 用户认证 配置域名跳转 配置apache的访问日志 配置静态文件缓存 配置防盗链 ...
- 风炫安全WEB安全学习第二十节课 反射型XSS讲解
风炫安全WEB安全学习第二十节课 反射型XSS讲解 反射性xss演示 原理讲解 如果一个应用程序使用动态页面向用户显示错误消息,就会造成一种常见的XSS漏洞.通常,该页面会使用一个包含消息文本的参数, ...
- 大白话5分钟带你走进人工智能-第二十节逻辑回归和Softmax多分类问题(5)
大白话5分钟带你走进人工智能-第二十节逻辑回归和Softmax多分类问题(5) 上一节中,我们讲 ...
- 聊聊高并发(十九)理解并发编程的几种"性" -- 可见性,有序性,原子性
这篇的主题本应该放在最初的几篇.讨论的是并发编程最基础的几个核心概念.可是这几个概念又牵扯到非常多的实际技术.比方Java内存模型.各种锁的实现,volatile的实现.原子变量等等,每个都可以展开写 ...
- [ExtJS5学习笔记]第二十节 Extjs5配合数组的push方法,动态创建并加载组件
本文地址:http://blog.csdn.net/sushengmiyan/article/details/39226773 官方例子:http://docs.sencha.com/extjs/5. ...
- 第二十节,使用RNN网络拟合回声信号序列
这一节使用TensorFlow中的函数搭建一个简单的RNN网络,使用一串随机的模拟数据作为原始信号,让RNN网络来拟合其对应的回声信号. 样本数据为一串随机的由0,1组成的数字,将其当成发射出去的一串 ...
- 第二十节:详细讲解String和StringBuffer和StringBuilder的使用
前言 在 Java中的字符串属于对象,那么Java 中提供了 String 类来创建和操作字符串,即是使用对象:因为String类修饰的字符一旦被创建就不可改变,所以当对字符串进行修改的时候,需要使用 ...
- 第二十节:Asp.Net Core WebApi生成在线文档
一. 基本概念 1.背景 使用 Web API 时,了解其各种方法对开发人员来说可能是一项挑战. Swagger 也称为OpenAPI,解决了为 Web API 生成有用文档和帮助页的问题. 它具有诸 ...
- 第二十节,基本数据类型,集合set、综合应用新数据更新老数据
基本数据类型,集合set.综合应用新数据更新老数据 创建两个字典新数据,更新原始数据,a为原始数据,b为新数据 1,分别获取到a字典和b字典的key(键),将两个字典的键分别转换成两个集合 2,找出a ...
随机推荐
- 【Python 13】分形树绘制1.0--五角星(turtle库)
1.案例描述 2.案例分析 引入绘制图形的turtle库,利用库中函数进行编程. 3.turtle库 没有显示的input()和output(),没有赋值语句.调用形式大部分如下: import tu ...
- velocity模板引擎 -- java.io.FileNotFoundException: velocity.log (Permission denied)
问题原因是velocity的日志框架导致(velocity是使用自己封装的日志框架记录日志的),velocity在初始化Logger时,如果没有读取到配置文件,则会使用默认的velocity.log做 ...
- 位(Bit)与字节(Byte)
字 word 字节 byte 位 bit 字长是指字的长度 1字=2字节(1 word = 2 byte) 1字节=8位(1 byte = 8bit) 一个字的字长为16 一个字节的字长是8 bps ...
- loadrunner迭代和并发的区别
转载: ZEE的回答: 用比喻的方式来回一下: 四车道的马路,如果只有四辆车并排走过就是并发: 如果四辆车排成一纵队走过就是迭代: 如果有100辆车排成25行依次走过就是并发加迭代. 在以上说法中,只 ...
- [SCOI2016]萌萌哒
Luogu P3295 mrclr两周前做的题让蒟蒻的我现在做? 第一眼组合计数,如果把数字相同的数位看作一个整体,除了第一位不能为零,剩下的每一位都有$0$~$9$十种. 设不同的位数为$x$,那么 ...
- Java Lucene入门
1.lucene版本:7.2.1 pom文件: <?xml version="1.0" encoding="UTF-8"?> <project ...
- 偶现bug如何处理?
请先允许我对此类bug进行吐槽,相信做测试的同学都碰见过这种bug! 我们在测试过程中经常会碰见一类很头疼的bug,就是偶现性的bug,所谓偶现性,是相对于必现而言,这类bug有些可以有重现路径,但是 ...
- keepalived--小白博客
一.HA集群中的相关术语 1.节点(node) 运行HA进程的一个独立主机,称为节点,节点是HA的核心组成部分,每个节点上运行着操作系统和高可用软件服务,在高可用集群中,节点有主次之分,分别称之为主节 ...
- Sql Server登录失败问题
1.启动SQL Server 2008 Management Studio,会看到 2. 里面有一个 身份验证.这个 身份验证 的下拉列表里面有两个选项: Windows 身份验证 和 SQL Ser ...
- asp.net core 中间件粗解
中间件 中间件在asp.net core中非常重要,它用来处理httpcontext.而httpcontext封装了请求和响应.也就是说,中间件是用来处理请求和响应的. 本质上,中间件被封装到了IAp ...