亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

MySQL 行鎖和表鎖的含義及區(qū)別詳解

 更新時(shí)間:2020年08月18日 09:42:22   作者:巔峰大詞典  
這篇文章主要介紹了MySQL 行鎖和表鎖的含義及區(qū)別詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

一、前言

對(duì)于行鎖和表鎖的含義區(qū)別,在面試中應(yīng)該是高頻出現(xiàn)的,我們應(yīng)該對(duì)MySQL中的鎖有一個(gè)系統(tǒng)的認(rèn)識(shí),更詳細(xì)的需要自行查閱資料,本篇為概括性的總結(jié)回答。

MySQL常用引擎有MyISAM和InnoDB,而InnoDB是mysql默認(rèn)的引擎。MyISAM不支持行鎖,而InnoDB支持行鎖和表鎖。

相對(duì)其他數(shù)據(jù)庫而言,MySQL的鎖機(jī)制比較簡(jiǎn)單,其最顯著的特點(diǎn)是不同的存儲(chǔ)引擎支持不同的鎖機(jī)制。

MySQL大致可歸納為以下3種鎖:

  • 表級(jí)鎖:開銷小,加鎖快;不會(huì)出現(xiàn)死鎖;鎖定粒度大,發(fā)生鎖沖突的概率最高,并發(fā)度最低。
  • 行級(jí)鎖:開銷大,加鎖慢;會(huì)出現(xiàn)死鎖;鎖定粒度最小,發(fā)生鎖沖突的概率最低,并發(fā)度也最高。
  • 頁面鎖:開銷和加鎖時(shí)間界于表鎖和行鎖之間;會(huì)出現(xiàn)死鎖;鎖定粒度界于表鎖和行鎖之間,并發(fā)度一般

如何加鎖?

MyISAM在執(zhí)行查詢語句(SELECT)前,會(huì)自動(dòng)給涉及的所有表加讀鎖,在執(zhí)行更新操作(UPDATE、DELETE、INSERT等)前,會(huì)自動(dòng)給涉及的表加寫鎖,這個(gè)過程并不需要用戶干預(yù),因此用戶一般不需要直接用LOCK TABLE命令給MyISAM表顯式加鎖。

顯式加鎖:

上共享鎖(讀鎖)的寫法:lock in share mode,例如:

select math from zje where math>60 lock in share mode;

上排它鎖(寫鎖)的寫法:for update,例如:

select math from zje where math >60 for update;

二、表鎖

不會(huì)出現(xiàn)死鎖,發(fā)生鎖沖突幾率高,并發(fā)低。

MyISAM引擎

MyISAM在執(zhí)行查詢語句(select)前,會(huì)自動(dòng)給涉及的所有表加讀鎖,在執(zhí)行增刪改操作前,會(huì)自動(dòng)給涉及的表加寫鎖。

MySQL的表級(jí)鎖有兩種模式:

  • 表共享讀鎖
  • 表獨(dú)占寫鎖

讀鎖會(huì)阻塞寫,寫鎖會(huì)阻塞讀和寫

  • 對(duì)MyISAM表的讀操作,不會(huì)阻塞其它進(jìn)程對(duì)同一表的讀請(qǐng)求,但會(huì)阻塞對(duì)同一表的寫請(qǐng)求。只有當(dāng)讀鎖釋放后,才會(huì)執(zhí)行其它進(jìn)程的寫操作。
  • 對(duì)MyISAM表的寫操作,會(huì)阻塞其它進(jìn)程對(duì)同一表的讀和寫操作,只有當(dāng)寫鎖釋放后,才會(huì)執(zhí)行其它進(jìn)程的讀寫操作。

MyISAM不適合做寫為主表的引擎,因?yàn)閷戞i后,其它線程不能做任何操作,大量的更新會(huì)使查詢很難得到鎖,從而造成永遠(yuǎn)阻塞

三、行鎖

會(huì)出現(xiàn)死鎖,發(fā)生鎖沖突幾率低,并發(fā)高。

在MySQL的InnoDB引擎支持行鎖,與Oracle不同,MySQL的行鎖是通過索引加載的,也就是說,行鎖是加在索引響應(yīng)的行上的,要是對(duì)應(yīng)的SQL語句沒有走索引,則會(huì)全表掃描,行鎖則無法實(shí)現(xiàn),取而代之的是表鎖,此時(shí)其它事務(wù)無法對(duì)當(dāng)前表進(jìn)行更新或插入操作。

CREATE TABLE `user` (
`name` VARCHAR(32) DEFAULT NULL,
`count` INT(11) DEFAULT NULL,
`id` INT(11) NOT NULL AUTO_INCREMENT,
PRIMARY KEY (`id`)
) ENGINE=INNODB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8

-- 這里,我們建一個(gè)user表,主鍵為id

 

-- A通過主鍵執(zhí)行插入操作,但事務(wù)未提交
update user set count=10 where id=1;
-- B在此時(shí)也執(zhí)行更新操作
update user set count=10 where id=2;
-- 由于是通過主鍵選中的,為行級(jí)鎖,A和B操作的不是同一行,B執(zhí)行的操作是可以執(zhí)行的

 

-- A通過name執(zhí)行插入操作,但事務(wù)未提交
update user set count=10 where name='xxx';
-- B在此時(shí)也執(zhí)行更新操作
update user set count=10 where id=2;
-- 由于是通過非主鍵或索引選中的,升級(jí)為為表級(jí)鎖,
-- B則無法對(duì)該表進(jìn)行更新或插入操作,只有當(dāng)A提交事務(wù)后,B才會(huì)成功執(zhí)行

for update

如果在一條select語句后加上for update,則查詢到的數(shù)據(jù)會(huì)被加上一條排它鎖,其它事務(wù)可以讀取,但不能進(jìn)行更新和插入操作

-- A用戶對(duì)id=1的記錄進(jìn)行加鎖
select * from user where id=1 for update;

-- B用戶無法對(duì)該記錄進(jìn)行操作
update user set count=10 where id=1;

-- A用戶commit以后則B用戶可以對(duì)該記錄進(jìn)行操作

行鎖的實(shí)現(xiàn)需要注意:

  • 行鎖必須有索引才能實(shí)現(xiàn),否則會(huì)自動(dòng)鎖全表,那么就不是行鎖了。
  • 兩個(gè)事務(wù)不能鎖同一個(gè)索引。
  • insert,delete,update在事務(wù)中都會(huì)自動(dòng)默認(rèn)加上排它鎖。

行鎖場(chǎng)景:

A用戶消費(fèi),service層先查詢?cè)撚脩舻馁~戶余額,若余額足夠,則進(jìn)行后續(xù)的扣款操作;這種情況查詢的時(shí)候應(yīng)該對(duì)該記錄進(jìn)行加鎖。

否則,B用戶在A用戶查詢后消費(fèi)前先一步將A用戶賬號(hào)上的錢轉(zhuǎn)走,而此時(shí)A用戶已經(jīng)進(jìn)行了用戶余額是否足夠的判斷,則可能會(huì)出現(xiàn)余額已經(jīng)不足但卻扣款成功的情況。

為了避免此情況,需要在A用戶操作該記錄的時(shí)候進(jìn)行for update加鎖

擴(kuò)展:間隙鎖

當(dāng)我們用范圍條件而不是相等條件檢索數(shù)據(jù),并請(qǐng)求共享或排他鎖時(shí),InnoDB會(huì)給符合條件的已有數(shù)據(jù)記錄的索引項(xiàng)加鎖;對(duì)于鍵值在條件范圍內(nèi)并不存在的記錄,叫做間隙

InnoDB也會(huì)對(duì)這個(gè)"間隙"加鎖,這種鎖機(jī)制就是所謂的間隙鎖

-- 用戶A
update user set count=8 where id>2 and id<6

-- 用戶B
update user set count=10 where id=5;

如果用戶A在進(jìn)行了上述操作后,事務(wù)還未提交,則B無法對(duì)2~6之間的記錄進(jìn)行更新或插入記錄,會(huì)阻塞,當(dāng)A將事務(wù)提交后,B的更新操作會(huì)執(zhí)行。

建議:

  • 盡可能讓所有數(shù)據(jù)檢索都通過索引來完成,避免無索引行鎖升級(jí)為表鎖
  • 合理設(shè)計(jì)索引,盡量縮小鎖的范圍
  • 盡可能減少索引條件,避免間隙鎖
  • 盡量控制事務(wù)大小,減少鎖定資源量和時(shí)間長(zhǎng)度

 到此這篇關(guān)于MySQL 行鎖和表鎖的含義及區(qū)別詳解的文章就介紹到這了,更多相關(guān)MySQL 行鎖和表鎖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • mysql實(shí)用操作之日期格式化例子

    mysql實(shí)用操作之日期格式化例子

    這篇文章主要給大家介紹了關(guān)于mysql實(shí)用操作之日期格式化的相關(guān)資料,DATE_FORMAT()函數(shù)用于格式化MySQL中的日期和時(shí)間字段,支持多種格式化選項(xiàng),需要的朋友可以參考下
    2024-11-11
  • MySQL 四種事務(wù)隔離級(jí)別詳解及對(duì)比

    MySQL 四種事務(wù)隔離級(jí)別詳解及對(duì)比

    這篇文章主要介紹了MySQL 四種事務(wù)隔離級(jí)別詳解及對(duì)比的相關(guān)資料,這里對(duì)Mysql 的基礎(chǔ)知識(shí)做了詳細(xì)介紹及對(duì)其事務(wù)隔離進(jìn)行了比較,需要的朋友可以參考下
    2016-12-12
  • mysql增量語句如何修改表的字段

    mysql增量語句如何修改表的字段

    本文介紹了如何在MySQL中實(shí)現(xiàn)數(shù)據(jù)庫的增量更新,確保多次執(zhí)行增量語句不會(huì)影響最終結(jié)果,通過使用存儲(chǔ)過程,解決了MySQL無法直接執(zhí)行塊語句的問題,文章詳細(xì)描述了增量添加、修改和刪除某個(gè)字段的操作,并總結(jié)了增量語句的編寫方法,類似于編寫冪等接口
    2024-12-12
  • MAC下Mysql5.7.10版本修改root密碼的方法

    MAC下Mysql5.7.10版本修改root密碼的方法

    這篇文章主要介紹了MAC下Mysql5.7.10版本修改root密碼的方法,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2017-03-03
  • 虛擬主機(jī)MySQL數(shù)據(jù)庫的備份與還原的方法

    虛擬主機(jī)MySQL數(shù)據(jù)庫的備份與還原的方法

    虛擬主機(jī)MySQL數(shù)據(jù)庫的備份與還原的方法...
    2007-07-07
  • 淺談MySQL使用笛卡爾積原理進(jìn)行多表查詢

    淺談MySQL使用笛卡爾積原理進(jìn)行多表查詢

    這篇文章主要介紹了淺談MySQL使用笛卡爾積原理進(jìn)行多表查詢, 文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • Linux安裝MySQL教程(二進(jìn)制分發(fā)版)

    Linux安裝MySQL教程(二進(jìn)制分發(fā)版)

    這篇文章主要為大家詳細(xì)介紹了Linux安裝MySQL教程,二進(jìn)制分發(fā)版,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-02-02
  • MySQL中一些優(yōu)化straight_join技巧

    MySQL中一些優(yōu)化straight_join技巧

    這篇文章主要介紹了MySQL中一些優(yōu)化straight_join技巧,作者通過用戶的實(shí)際案例分析,需要的朋友可以參考下
    2015-05-05
  • mysql間隙鎖加鎖11個(gè)規(guī)則(案例分析)

    mysql間隙鎖加鎖11個(gè)規(guī)則(案例分析)

    這篇文章主要介紹了mysql間隙鎖加鎖11個(gè)規(guī)則?,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-03-03
  • 淺談Mysql時(shí)間的存儲(chǔ)?datetime還是時(shí)間戳timestamp

    淺談Mysql時(shí)間的存儲(chǔ)?datetime還是時(shí)間戳timestamp

    本文主要介紹了淺談Mysql時(shí)間的存儲(chǔ)?datetime還是時(shí)間戳timestamp,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07

最新評(píng)論