国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Select For Update加鎖,懵圈了!

數據庫 MySQL
Select...for update在MySQL中,是一種悲觀鎖的用法,一般情況下,會鎖住一行數據,但如果沒有使用正確的話,也會把整張表鎖住,導致SQL性能急劇下降。其實,我之前也在實際項目中試過用For update關鍵字加行鎖,比如:積分兌換禮品的功能。今天就跟大家一起聊聊Select...for update這個話題,希望對你會有所幫助。

前言

前幾天,知識星球中的一個小伙伴,問了我一個問題:在MySQL中,事務A中使用select...for update where id=1鎖住了,某一條數據,事務還沒提交,

此時,事務B中去用select ... where id=1查詢那條數據,會阻塞等待嗎?

其實select...for update在MySQL中,是一種悲觀鎖的用法,一般情況下,會鎖住一行數據,但如果沒有使用正確的話,也會把整張表鎖住,導致SQL性能急劇下降。

其實,我之前也在實際項目中試過用for update關鍵字加行鎖,比如:積分兌換禮品的功能。

今天就跟大家一起聊聊select...for update這個話題,希望對你會有所幫助。

1、要什么要用行鎖?

假如現在有這樣一種業務場景:用戶A給你轉賬了2000元,此時剛好,用戶B也給你轉賬了3000元,而你的賬戶初始化金額是1000元。

用戶A的請求,先查詢出money,然后給money加上2000,在事務1中會執行下面這條sql:

update account set money=#{money}
where id=123;

同事,用戶B的請求,也是先查詢出money,然后給money加上3000,在事務2中執行下面這條sql:

update account set money=#{money}
where id=123;

這兩條sql執行成功之后,你的money可能是:3000、4000、6000,這三種情況中的一種。

你之前的想法是,用戶A和用戶B總共給你轉賬5000,最終你賬戶的錢應該是6000才對,3000和4000是怎么來的?

假如事務1在執行update語句的過程中,事務2同時也在執行update語句。

事務1中查詢到money是1000,此外事務2也查詢到money是1000。

如果事務1先執行update語句,事務2后執行update語句,第一次update的3000,會被后面的4000覆蓋掉,最終結果為4000。

如果事務2先執行update語句,事務1后執行update語句,第一次update的4000,會被后面的3000覆蓋掉,最終結果為3000。

這兩種情況都產生了嚴重的數據問題。

我們需要有某種機制,保證計算金額后事務1和事務2要順序執行,不要一起執行。

這就需要加鎖了。

目前MySQL中使用比較多的有:表鎖、行鎖和間隙鎖。

我們這個業務場景,非常時候使用行鎖。

在事務1執行update語句的過程中,先要把某一行數據鎖住,此時,其他的事務必須等待事務1執行完,提交了事務,才能獲取那一行的數據。

在MySQL中是通過select...for update語句來實現的行鎖的功能。

但如果你在實際工作中使用不正確,也容易把整張表鎖住,嚴重影響性能。

select...where...for update語句的用法是否正確,跟where條件中的參數有很大的關系。

不信我們一起看看下面這幾種情況。

假如user表現在有這樣的數據庫,數據庫的版本是:8.0.21。

創建的索引如下:

其中id是主鍵字段,code是唯一索引字段,name是普通索引字段,其他的都是普通字段。

2、主鍵

當where條件中用的是數據庫主鍵時。

例如開啟一個事務1,在事務中更新id=1的用戶的年齡:

begin;
select * from user where id=1 for update;
update user set age=22 where id=1;

where條件中的id是數據庫的主鍵,并且使用for update關鍵字,加了一個行鎖,這個事務沒有commit。

此時,開啟了另外一個事務2,也更新id=1的用戶的年齡:

begin;
update user set age=23 where id=1;
commit;

在執行事務2的sql語句的過程中,會一直等待事務1釋放鎖。

如果事務1一直都不釋放行鎖,事務2最后會報下面這個異常:

如果此時開始一個事務3,更新id=2的用戶的年齡:

begin;
update user set age=23 where id=2;
commit;

執行結果如下:

由于事務3中更新的另外一行數據,因此可以執行成功。

說明使用for update關鍵字,鎖住了主鍵id=1的那一行數據,對其他行的數據并沒有影響。

3、唯一索引

當where條件用的數據庫唯一索引時。

開啟一個事務1,在事務中更新code=101的用戶的年齡:

begin;
select * from user where code='101' for update;
update user set age=22 where code='101';

where條件中的code是數據庫的唯一索引,并且使用for update關鍵字,加了一個行鎖,這個事務沒有commit。

此時,開啟了另外一個事務2,也更新code=101的用戶的年齡:

begin;
update user set age=23 where code='101';
commit;

執行結果跟主鍵的情況是一樣的。

4、普通索引

當where條件用的數據庫普通索引時。

開啟一個事務1,在事務中更新name=周星馳的用戶的年齡:

begin;
select * from user where name='周星馳' for update;
update user set age=22 where name='周星馳';

where條件中的name是數據庫的普通索引,并且使用for update關鍵字,加了一個行鎖,這個事務沒有commit。

此時,開啟了另外一個事務2,也更新name=周星馳的用戶的年齡:

begin;
update user set age=23 where name='周星馳';
commit;

執行結果跟主鍵的情況也是一樣的。

5、主鍵范圍

當where條件用的數據庫主鍵范圍時。

開啟一個事務1,在事務中更新id in (1,2)的用戶的年齡:

begin;
select * from user where id in (1,2) for update;
update user set age=22 where id in (1,2);

where條件中的id是數據庫的主鍵范圍,并且使用for update關鍵字,加了多個行鎖,這個事務沒有commit。

此時,開啟了另外一個事務2,也更新id=1的用戶的年齡:

begin;
update user set age=23 where id=1;
commit;

執行結果跟主鍵的情況也是一樣的。

此時,開啟了另外一個事務2,也更新id=2的用戶的年齡:

begin;
update user set age=23 where id=2;
commit;

執行結果跟主鍵的情況也是一樣的。

6、普通字段

當where條件用的數據庫普通字段時。

該字段既不是主鍵,也不是索引。

開啟一個事務1,在事務中更新age=22的用戶的年齡:

begin;
select * from user where age=22 for update;
update user set age=22 where age=22 ;

where條件中的age是數據庫的普通字段,并且使用for update關鍵字,加的是表鎖,這個事務沒有commit。

此時,開啟了另外一個事務2,也更新age=22的用戶的年齡:

begin;
update user set age=23 where age=22 ;
commit;

此時,執行事務2時,會一直阻塞等待事務1釋放鎖。

調整一下sql條件,查詢條件改成age=23:

begin;
update user set age=23 where age=23 ;
commit;

此時,行事務3時,也會一直阻塞等待事務1釋放鎖。

也就是說,在for update語句中,使用普通字段作為查詢條件時,加的不是行鎖。

那么,到底是什么鎖呢?

開啟一個事務4,在事務中更新age=22的用戶的年齡:

begin;
select * from user where age=23 for update;
update user set age=22 where age=23 ;

嘗試insert一條age=22的新數據:

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (6, '105', 22, '蘇三說技術', 173, '武漢', NULL, NULL);

最后發現insert失敗了。

嘗試insert一條age=23的新數據:

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (6, '105', 23, '蘇三說技術', 173, '武漢', NULL, NULL);

最后發現insert也失敗了。

而把age改成21重新insert:

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (6, '105', 21, '蘇三說技術', 173, '武漢', NULL, NULL);

卻insert成功了:

意不意外?驚不驚喜?

說明這種情況下,加的不是行鎖,也不是表鎖,而是間隙鎖,鎖定的范圍是age從【22~∞】。

7、空數據

當where條件查詢的數據不存在時,會發生什么呢?

開啟一個事務1,在事務中更新id=66的用戶的年齡:

begin;
select * from user where id=66 for update;

這條數據是不存在的。

此時,開啟了另外一個事務2,也更新id=66的用戶的年齡:

begin;
update user set age=23 where id=66 ;
commit;

執行結果:

執行成功了,說明這種情況沒有加鎖?

不繼續往下看。

開啟事務3,insert一條age=21的數據:

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (5, '104', 21, '蘇三說技術', 173, '武漢', NULL, NULL);

結果insert也失敗了:

說明用for update關鍵字,通過主鍵查詢空數據時,是加了鎖的,目前得知不是行鎖。

是表鎖?

假如insert一條age=65的數據:

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (6, '106', 65, '蘇三說技術', 173, '武漢', NULL, NULL);

發現insert失敗了:

改成insert一條age=21的數據呢?

INSERT INTO `sue`.`user`(`id`, `code`, `age`, `name`, `height`, `address`, `phone`, `encrypt_phone`) VALUES (8, '108', 21, '蘇三說技術', 173, '武漢', NULL, NULL);

結果insert成功了:

說明用for update關鍵字,通過主鍵查詢空數據時,加的不是表鎖,而是間隙鎖。

總結

最后給大家總結一下select...for update加鎖的情況:

  • 主鍵字段:加行鎖。
  • 唯一索引字段:加行鎖。
  • 普通索引字段:加間隙鎖。
  • 主鍵范圍:加多個行鎖。
  • 唯一索引范圍,加多個行鎖。
  • 普通字段:加表鎖。
  1. 查詢空數據:加間鎖。

如果事務1加了行鎖,一直沒有釋放鎖,事務2操作相同行的數據時,會一直等待直到超時。

如果事務1加了表鎖,一直沒有釋放鎖,事務2不管操作的是哪一行數據,都會一直等待直到超時。

此外,有些小伙伴,可能會好奇,直接執行update語句,也會加行鎖,為什么還需要使用for update關鍵字加行鎖呢?

答:for update關鍵字是加在select語句中的,它從查到那行數據開始,直到事務提交,整個過程中都會加鎖。

而直接執行update語句,是在更新數據的時候加鎖,二者有本質的區別。

責任編輯:姜華 來源: 蘇三說技術
相關推薦

2023-10-25 08:21:15

悲觀鎖MySQL

2022-04-10 18:10:24

CURD鏈表

2021-04-21 08:54:49

Go語言程序

2025-10-31 01:12:00

2017-12-06 09:00:14

2021-08-26 06:57:53

零拷貝技術磁盤

2020-06-22 08:50:27

Spring AOP代理

2023-01-11 09:56:41

索引SQL

2021-04-06 06:23:18

MVCC并發事務

2020-10-13 16:30:31

語言鏈表數組

2018-12-28 14:47:34

大數據云計算數據庫

2020-11-09 08:51:24

6G衛星

2023-12-18 08:03:56

并發編程Java

2021-09-01 18:38:59

Goselectdefault

2024-06-14 09:27:00

2024-03-06 08:18:22

語句GreatSQL

2025-04-14 10:30:00

IP地址API定位互聯網

2019-09-27 09:13:55

Redis內存機制

2018-02-07 07:00:09

2020-07-30 07:47:32

互聯網
點贊
收藏

51CTO技術棧公眾號

六月婷婷激情网| 亚洲欧洲成人自拍| 亚洲精品免费视频| 日韩免费av一区二区三区| 日本一区二区三区播放| 99精品视频中文字幕| 日本久久久久久久| 日本一卡二卡四卡精品| 日韩激情图片| 亚洲国内高清视频| 免费av片在线观看一道本| 福利电影一区| 精品久久久久久久久久久久包黑料| 日韩av黄色网址| 欧美韩日高清| 亚洲天堂av图片| 国外av网站| 久久―日本道色综合久久| 欧美激情视频在线| 久久精品66| 日韩成人中文字幕| 九色porny丨入口在线| 日韩毛片高清在线播放| 99re6在线视频| 亚洲综合一区二区三区| 久久综合色播| 日韩理论在线观看| 亚洲精品视频在线免费| 精品久久久久久久久久久久久 | 日韩精品中文字幕在线一区| 男女h黄动漫啪啪无遮挡软件| 欧美韩国日本| 色综合亚洲欧洲| 日韩av资源在线| 国产亚洲在线| 国产欧美日韩综合精品二区| 99精品视频在线免费播放| 精品国产网站在线观看| 综合网插菊花| 精东粉嫩av免费一区二区三区| 午夜av一区二区三区| 国产精品mp4| jlzzjlzz欧美| 欧美激情一区二区三区在线| 看一级黄色录像| 久久高清免费观看| 国产自摸综合网| 亚洲深夜视频| 日韩女优av电影在线观看| 久久婷婷国产91天堂综合精品| 久久蜜桃精品| 亚洲精品日韩激情在线电影| 美国av一区二区| 伊人发布在线| 精品日韩av一区二区| 一级毛片视频在线| 色综合久久悠悠| 国产一区在线看| 亚洲精品一区二区毛豆| 亚洲男人电影天堂| 搜成人激情视频| 蜜桃麻豆www久久国产精品| 久久九九久久九九| 超黄网站在线观看| 国产精品免费电影| 中文欧美字幕免费| 日本动漫理论片在线观看网站| 国产成人久久久| 亚洲欧美日韩中文字幕一区二区三区 | 亚洲综合三区| 欧洲美女少妇精品| 亚洲伊人一本大道中文字幕| 亚洲欧美日韩系列| japansex久久高清精品| 欧美日韩三区四区| 欧美一区二区三区播放老司机| 精品91视频| 国产三级视频在线看| 99porn视频在线| 久久精品男人天堂av| 91se在线| 国产传媒一区| 国产精品综合一区二区三区| 欧美午夜aaaaaa免费视频| 欧美高清在线一区| 青青国产在线| 国产亚洲精品久久久优势 | 精品国产丝袜高跟鞋| 精品国产乱码久久久久久影片| 午夜精品影视国产一区在线麻豆| 久久99精品视频一区97| 香蕉久久精品| 国产在线观看不卡| **欧美日韩在线观看| 亚洲精品电影在线| 鲁大师精品99久久久| 中文字幕日韩精品久久| 一本大道久久a久久综合婷婷| 五月激情久久| 一本色道久久综合狠狠躁篇怎么玩 | 亚洲一区自拍| 超碰在线公开| 国产综合av在线| 成人欧美一区二区三区小说| 欧美一区亚洲| 91在线直播| 一区高清视频| 91黄色8090| 欧美日韩国产bt| 久久精品噜噜噜成人av农村| 色播一区二区| 成人在线直播| 亚洲高清福利| 2019中文字幕在线视频| 爱草tv视频在线观看992| 欧美高清视频免费观看| 99精品国产高清一区二区麻豆| 欧洲美女免费图片一区| 天天亚洲美女在线视频| 日本中文字幕不卡| 中文在线а√在线8| 白天操夜夜操| 日韩一级免费在线观看| 欧美男人的天堂| 日韩av电影在线免费播放| 日韩理论片久久| 日韩女同互慰一区二区| 精品久久久久久久久久久| 不卡av在线免费观看| 国产成人精品网址| 一区在线视频| 青青草97国产精品麻豆| 欧美国产不卡| 少妇精品导航| 国模大尺度视频一区二区| 成人做爰免费视频免费看| 另类图片综合电影| 久久久精品一区二区毛片免费看| 成人ww免费完整版在线观看| 成人拍拍拍在线观看| 欧美色图色综合| 羞羞小视频视频| 欧美日韩系列| 法国空姐在线观看免费| 国产在线无码精品| 精品久久久久久久免费人妻| 亚洲另类第一页| 欧美 日韩 国产精品| 久久综合九色综合网站| 日韩一区免费观看| 国产精品免费不| 欧美日韩网址| 欧美日韩1区| 中国av一区二区三区| 日韩午夜在线影院| 成人免费视频在线观看超级碰| 成人黄色影片在线| 久久精品夜色噜噜亚洲a∨| 亚洲美女偷拍久久| 精品露脸国产偷人在视频| 午夜影院在线观看国产主播| 深夜福利免费在线观看| 精品国偷自产国产一区| 韩国19禁主播vip福利视频| 欧美另类老肥妇| 成人欧美一区二区三区视频xxx| 欧美3p视频| 日韩综合一区二区| 欧美在线不卡视频| 亚洲直播在线一区| 三级黄色网址| 大奶在线精品| 成人性生交大片免费| 亚洲成av人在线观看| 欧美大肥婆大肥bbbbb| 精品久久久久久一区| 99热成人精品热久久66| 日韩精品视频一区二区三区| wwwwxxxx日韩| 在线视频毛片| 亚洲国产精品免费视频| 亚洲国产综合在线看不卡| 久久精品国产999大香线蕉| 91麻豆精品国产91久久久久久久久 | 一区二区不卡在线视频 午夜欧美不卡在| 欧美成人精品福利| 久精品免费视频| 在线国产三级| 亚洲国产欧美国产第一区| av在线不卡电影| 亚洲成人动漫精品| 欧美大片免费观看| 欧美成人蜜桃| 全部孕妇毛片丰满孕妇孕| a级片免费在线观看| 99亚洲伊人久久精品影院红桃| 欧美一级片在线观看| av电影一区二区三区| 678在线观看视频| 国产精品视频|