日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

第五节:框架前期准备篇之锁机制处理并发

發(fā)布時(shí)間:2023/12/10 编程问答 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 第五节:框架前期准备篇之锁机制处理并发 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

一. 簡(jiǎn)介

(一). 在處理并發(fā)的這個(gè)問(wèn)題上,鎖大致分為兩類:悲觀鎖和樂(lè)觀鎖。

  1.? 悲觀鎖:悲觀的認(rèn)為每次去拿數(shù)據(jù)的時(shí)候都會(huì)被別人修改,所以每次在拿數(shù)據(jù)的時(shí)候都會(huì)“上鎖”,操作完成之后再“解鎖”。 在數(shù)據(jù)加鎖期間,其他人(其他線程)如果來(lái)拿數(shù)據(jù)就會(huì)等待,直到去掉鎖。數(shù)據(jù)庫(kù)層次的悲觀鎖有“表鎖”、“行鎖”等。

注:EF默認(rèn)不支持悲觀鎖,只能通過(guò)EF調(diào)用SQL語(yǔ)句。

  2.? 樂(lè)觀鎖:樂(lè)觀的認(rèn)為該條數(shù)據(jù)不會(huì)被占用,自己先占了再說(shuō),占完了之后再看看是不是占上了,如果沒(méi)占上就是操作失敗,提示給用戶。

  3.? 兩種鎖進(jìn)行對(duì)比:悲觀鎖使用的體驗(yàn)更好,但是對(duì)系統(tǒng)性能的影響大,只適合并發(fā)量不大的場(chǎng)合。 樂(lè)觀鎖適用于“寫少讀多”的情況下,加大了系統(tǒng)的整個(gè)吞吐量,但是“樂(lè)觀鎖可能失敗”給用戶的體驗(yàn)很不好。

注:兩種鎖各有利弊,至于怎么取舍,根據(jù)實(shí)際業(yè)務(wù)場(chǎng)景來(lái)進(jìn)行。

(二). 模擬一個(gè)搶單的業(yè)務(wù)場(chǎng)景

  一個(gè)乘客發(fā)了一個(gè)打車訂單,很多司機(jī)去搶這個(gè)訂單,執(zhí)行的業(yè)務(wù)簡(jiǎn)單點(diǎn)來(lái)說(shuō)是,先select出這條數(shù)據(jù),然后update這個(gè)條數(shù)據(jù)中的driveName字段為自己的名字,但是現(xiàn)在會(huì)有這么

?

一種現(xiàn)象,同時(shí)select出這條訂單,先后更新driveName這個(gè)字段,先搶到訂單的乘客會(huì)發(fā)現(xiàn)最后訂單沒(méi)了,實(shí)際上是數(shù)據(jù)庫(kù)中Update第二次的操作覆蓋了第一次的操作了,這就是

?

并發(fā)操作帶來(lái)的尷尬場(chǎng)景。

?

?

二. 悲觀鎖

?1. 數(shù)據(jù)準(zhǔn)備

  新建數(shù)據(jù)庫(kù)【LockDemoDB】,新建訂單表OrderInfor,包括字段有:id、userName(乘客姓名)、destination(訂單信息)、driverName(搶單司機(jī)的姓名)、isRobbed(該訂單是否被搶, 0代表未被搶,1代表已被搶 ),事先插入一條數(shù)據(jù)用于測(cè)試對(duì)應(yīng)的字段分別為: 1,? ypf,? 去北京,? "",? 0

如下圖:

?2. 原理

  開(kāi)啟事務(wù),利用排它鎖和行鎖將該條數(shù)據(jù)鎖住,其他線程如果要訪問(wèn),必須得該線程提交完事務(wù),鎖釋放后才能使用,下面分享兩種寫法:ADO.NET寫法 和 EF調(diào)用SQL語(yǔ)句寫法。

大致流程:

  ①:查詢id為1的數(shù)據(jù),如果不存在,則停止業(yè)務(wù);如果存在,繼續(xù)往下執(zhí)行。

  ②:查詢isRobbed字段的值,如果為1,代表該訂單已經(jīng)剛被人搶了,然后輸出driverName的值,即代表被誰(shuí)搶了;如果為0,代表該訂單尚未被搶,繼續(xù)往下執(zhí)行。

  ③:執(zhí)行Update操作,進(jìn)行事務(wù)提交,這期間別的線程是不能訪問(wèn)的。

  ④:提交完事務(wù)后,鎖被釋放,其它線程得以繼續(xù)訪問(wèn)。

?ADO.NET寫法:

1 {2 Console.WriteLine("司機(jī)您好,請(qǐng)輸入您的名字");3 string driverName = Console.ReadLine();4 string connstr = ConfigurationManager.ConnectionStrings["connstr"].ConnectionString;5 using (SqlConnection conn = new SqlConnection(connstr))6 {7 conn.Open();8 using (var tx = conn.BeginTransaction())9 { 10 try 11 { 12 Console.WriteLine("開(kāi)始查詢"); 13 using (var selectCmd = conn.CreateCommand()) 14 { 15 selectCmd.Transaction = tx; 16 //排它鎖和行鎖,針對(duì)訪問(wèn)線程鎖住該行,不能繼續(xù)往下執(zhí)行,只有事務(wù)提交完,其他線程才能訪問(wèn) 17 selectCmd.CommandText = "select * from OrderInfor with(xlock,ROWLOCK) where id=1"; 18 using (var reader = selectCmd.ExecuteReader()) 19 { 20 if (!reader.Read()) 21 { 22 Console.WriteLine("沒(méi)有id為1的訂單"); 23 return; 24 } 25 string dName = null; 26 string isRobbed = null; 27 if (!reader.IsDBNull(reader.GetOrdinal("driverName"))) 28 { 29 dName = reader.GetString(reader.GetOrdinal("driverName")); 30 } 31 if (!reader.IsDBNull(reader.GetOrdinal("isRobbed"))) 32 { 33 isRobbed = reader.GetString(reader.GetOrdinal("isRobbed")); 34 } 35 36 //表示該訂單已經(jīng)被搶了 37 if (isRobbed == "1" && !string.IsNullOrEmpty(dName)) 38 { 39 if (driverName == dName) 40 { 41 Console.WriteLine("該訂單早已經(jīng)被我搶了"); 42 } 43 else 44 { 45 Console.WriteLine($"該訂單早已經(jīng)被司機(jī)【{dName}】搶了"); 46 } 47 //不再往下執(zhí)行 48 Console.ReadKey(); 49 return; 50 } 51 } 52 Console.WriteLine("查詢完成,開(kāi)始執(zhí)行update操作"); 53 using (var updateCmd = conn.CreateCommand()) 54 { 55 updateCmd.Transaction = tx; 56 updateCmd.CommandText = "Update OrderInfor set driverName=@driverName,isRobbed=@isRobbed where id=1"; 57 updateCmd.Parameters.Add(new SqlParameter("@driverName", driverName)); 58 updateCmd.Parameters.Add(new SqlParameter("@isRobbed", "1")); 59 updateCmd.ExecuteNonQuery(); 60 } 61 Console.WriteLine("結(jié)束update操作"); 62 Console.WriteLine("按任意鍵進(jìn)行事務(wù)提交"); 63 Console.ReadKey(); 64 } 65 tx.Commit(); 66 Console.WriteLine("事務(wù)提交成功"); 67 } 68 catch (Exception ex) 69 { 70 Console.WriteLine(ex); 71 tx.Rollback(); 72 } 73 } 74 } 75 }

EF調(diào)用SQL語(yǔ)句寫法:

1 {2 Console.WriteLine("司機(jī)您好,請(qǐng)輸入您的名字");3 string driverName = Console.ReadLine();4 using (LockDemoDBEntities1 ctx = new LockDemoDBEntities1())5 using (var tx = ctx.Database.BeginTransaction())6 {7 Console.WriteLine("開(kāi)始查詢");8 //一定要遍歷一下 SqlQuery 的返回值才會(huì)真正執(zhí)行 SQL 9 //排它鎖和行鎖,針對(duì)訪問(wèn)線程鎖住該行,不能繼續(xù)往下執(zhí)行,只有事務(wù)提交完,其他線程才能訪問(wèn) 10 var orderInfor = ctx.Database.SqlQuery<OrderInfor>("select * from OrderInfor with(xlock,ROWLOCK) where id=1").Single(); 11 12 //表示該訂單已經(jīng)被搶了 13 if (orderInfor.isRobbed == "1" && !string.IsNullOrEmpty(orderInfor.driverName)) 14 { 15 if (driverName == orderInfor.driverName) 16 { 17 Console.WriteLine("該訂單早已經(jīng)被我搶了"); 18 } 19 else 20 { 21 Console.WriteLine($"該訂單早已經(jīng)被司機(jī)【{orderInfor.driverName}】搶了"); 22 } 23 //不再往下執(zhí)行 24 Console.ReadKey(); 25 return; 26 } 27 28 Console.WriteLine("查詢完成,開(kāi)始執(zhí)行update操作"); 29 ctx.Database.ExecuteSqlCommand("Update OrderInfor set driverName={0},isRobbed={1} where id=1", driverName, "1"); 30 Console.WriteLine("結(jié)束update操作"); 31 Console.WriteLine("按任意鍵進(jìn)行事務(wù)提交"); 32 Console.ReadKey(); 33 try 34 { 35 tx.Commit(); 36 } 37 catch (Exception ex) 38 { 39 Console.WriteLine(ex); 40 tx.Rollback(); 41 } 42 } 43 }

?結(jié)果分析:

?  ①:線程1進(jìn)入,查詢完畢,尚未進(jìn)行事務(wù)提交。

  ②:線程2進(jìn)入,被鎖住,無(wú)法繼續(xù)往下進(jìn)行操作。

  ③:線程1進(jìn)行事務(wù)提交,線程1執(zhí)行成功的同時(shí),線程2提示該訂單已經(jīng)被xx搶了。

?

?

三. 樂(lè)觀鎖

?1. 數(shù)據(jù)準(zhǔn)備

  新建訂單表OrderInfor2,包括基礎(chǔ)字段有:id、userName(乘客姓名)、destination(訂單信息)、driverName(搶單司機(jī)的姓名)、isRobbed(該訂單是否被搶, 0代表未被搶,1代表已被搶 ),?新增字段:rowversion字段, 類型為timestamp,對(duì)應(yīng)的實(shí)體類型為byte[],?事先插入一條數(shù)據(jù)用于測(cè)試對(duì)應(yīng)的字段分別為: 1, ypf, 去北京, "", 0 。

PS:凡是對(duì)該條數(shù)據(jù)進(jìn)行過(guò)update操作,rowversion字段的值都會(huì)發(fā)生變化。

?2. 原理

? ? 這里提供兩種思路,分別是:原生的SQL語(yǔ)句(這里通過(guò)EF調(diào)用) 和 EF默認(rèn)的樂(lè)觀鎖模式。

(1). 原生的SQL語(yǔ)句:

  ①:查出該條訂單的記錄,包括rowversion字段。

  ②:把該rowversion字段作為update操作where的一個(gè)條件,執(zhí)行更新操作。

  ③:看受影響的行數(shù),如果受影響的行數(shù)為0,表示該條數(shù)據(jù)在你執(zhí)行更新操作前已經(jīng)被人改過(guò)了,這個(gè)時(shí)候通常提示用戶“更新失敗”;如果受影響的行數(shù)為1,則表示沒(méi)被修改過(guò),提示用戶“更新成功”。

分享代碼:

1 {2 try3 {4 Console.WriteLine("司機(jī)您好,請(qǐng)輸入您的名字");5 string driverName = Console.ReadLine();6 using (LockDemoDBEntities1 ctx = new LockDemoDBEntities1())7 {8 Console.WriteLine("開(kāi)始查詢");9 //一定要遍歷一下 SqlQuery 的返回值才會(huì)真正執(zhí)行 SQL 10 var orderInfor = ctx.Database.SqlQuery<OrderInfor2>("select * from OrderInfor2 where id=1").Single(); 11 12 //表示該訂單已經(jīng)被搶了 13 if (orderInfor.isRobbed == "1" && !string.IsNullOrEmpty(orderInfor.driverName)) 14 { 15 if (driverName == orderInfor.driverName) 16 { 17 Console.WriteLine("該訂單早已經(jīng)被我搶了"); 18 } 19 else 20 { 21 Console.WriteLine($"該訂單早已經(jīng)被司機(jī)【{orderInfor.driverName}】搶了"); 22 } 23 //不在往下執(zhí)行 24 Console.ReadKey(); 25 return; 26 } 27 28 Console.WriteLine("查詢完成,按任意鍵進(jìn)行搶單"); 29 Console.ReadKey(); 30 Console.WriteLine("正在搶單中。。。。。"); 31 //休眠3s,模擬高并發(fā)搶單 32 Thread.Sleep(3000); 33 int affectRows = ctx.Database.ExecuteSqlCommand("Update OrderInfor2 set driverName={0},isRobbed={1} where id=1 and rowversion={2}", driverName, "1", orderInfor.rowversion); 34 if (affectRows == 0) 35 { 36 Console.WriteLine("搶單失敗"); 37 } 38 else if (affectRows == 1) 39 { 40 Console.WriteLine("搶單成功"); 41 } 42 else 43 { 44 Console.WriteLine("見(jiàn)鬼了"); 45 } 46 } 47 } 48 catch (Exception ex) 49 { 50 Console.WriteLine("失敗了"); 51 Console.WriteLine(ex.Message); 52 throw; 53 } 54 }

(2). EF默認(rèn)的樂(lè)觀鎖模式

  a. DBFirst模式:在Edmx模型上給該字段的并發(fā)模式設(shè)置為fixed(默認(rèn)為None),這樣該表中所有字段都監(jiān)控并發(fā)。如果不想監(jiān)視所有列(在不添加RowVersion的情況下),只需在Edmx模型是給特定的字段的并發(fā)模式設(shè)置為fixed,這樣只有被設(shè)置的字段被監(jiān)測(cè)并發(fā)。

  b. CodeFirst下的Fluent API下的配置:

      全局配置:Property(e => e.RowVersion).IsRowVersion();

      單獨(dú)字段配置:Property(p => p.xxxx).IsConcurrencyToken();

  c. CodeFirst下的DataAnnotation下的配置:rowversion屬性加上特性1702220125,這樣該表中所有字段都監(jiān)控并發(fā)。如果不想監(jiān)視所有列(在不添加RowVersion的情況下),?只需給特定的字段加上特性 [ConcurrencyCheck],這樣只有被設(shè)置的字段被監(jiān)測(cè)并發(fā)。

原理:通過(guò)DbUpdateConcurrencyException監(jiān)測(cè)該條數(shù)據(jù)是否被改過(guò),改過(guò)就拋異常。

分享代碼:

1 Console.WriteLine("司機(jī)您好,請(qǐng)輸入您的名字");2 string driverName = Console.ReadLine();3 using (LockDemoDBEntities1 ctx = new LockDemoDBEntities1())4 {5 Console.WriteLine("開(kāi)始查詢");6 7 var orderInfor = ctx.OrderInfor2.Where(u => u.id == "1").FirstOrDefault();8 9 //表示該訂單已經(jīng)被搶了 10 if (orderInfor.isRobbed == "1" && !string.IsNullOrEmpty(orderInfor.driverName)) 11 { 12 if (driverName == orderInfor.driverName) 13 { 14 Console.WriteLine("該訂單早已經(jīng)被我搶了"); 15 } 16 else 17 { 18 Console.WriteLine($"該訂單早已經(jīng)被司機(jī)【{orderInfor.driverName}】搶了"); 19 } 20 //不在往下執(zhí)行 21 Console.ReadKey(); 22 return; 23 } 24 25 Console.WriteLine("查詢完成,按任意鍵進(jìn)行搶單"); 26 Console.ReadKey(); 27 Console.WriteLine("正在搶單中。。。。。"); 28 //休眠3s,模擬高并發(fā)搶單 29 Thread.Sleep(3000); 30 31 //下面執(zhí)行更新操作 32 orderInfor.driverName = driverName; 33 orderInfor.isRobbed = "1"; 34 try 35 { 36 ctx.SaveChanges(); 37 Console.WriteLine("搶單成功"); 38 } 39 catch (DbUpdateConcurrencyException) 40 { 41 Console.WriteLine("搶單失敗"); 42 } 43 }

3. 結(jié)果分析

?  ①:線程1 和 線程2,同時(shí)執(zhí)行且均查詢完畢,等待點(diǎn)擊按鈕進(jìn)行搶單。

  ②:先點(diǎn)擊線程1,然后點(diǎn)擊線程2,發(fā)現(xiàn)線程1搶單成功,線程2搶單失敗,證明線程2在搶單的時(shí)候,監(jiān)測(cè)到該數(shù)據(jù)已經(jīng)被改動(dòng)了。

?

?

四. 數(shù)據(jù)庫(kù)鎖詳解

?

詳見(jiàn):https://www.cnblogs.com/yaopengfei/p/9762267.html

?

?

?

?

?

?

?

!

  • 作???????者 :?Yaopengfei(姚鵬飛)
  • 博客地址 :?http://www.cnblogs.com/yaopengfei/
  • 聲?????明1 : 本人才疏學(xué)淺,用郭德綱的話說(shuō)“我是一個(gè)小學(xué)生”,如有錯(cuò)誤,歡迎討論,請(qǐng)勿謾罵^_^。
  • 聲?????明2 : 原創(chuàng)博客請(qǐng)?jiān)谵D(zhuǎn)載時(shí)保留原文鏈接或在文章開(kāi)頭加上本人博客地址,否則保留追究法律責(zé)任的權(quán)利。

總結(jié)

以上是生活随笔為你收集整理的第五节:框架前期准备篇之锁机制处理并发的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。