MySQL行鎖和表鎖的含義及區別

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

 

如何加鎖?

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

顯式加鎖:

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

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

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

select math from zjk where math >60 for update;

 表鎖

不會出現死鎖,發生鎖衝突機率高,併發低。

MyISAM引擎

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

MySQL的表級鎖有兩種模式:

  • 表共享讀鎖

  • 表獨佔寫鎖

讀鎖會阻塞寫,寫鎖會阻塞讀和寫

  • 對MyISAM表的讀操作,不會阻塞其它進程對同一表的讀請求,但會阻塞對同一表的寫請求。只有當讀鎖釋放後,纔會執行其它進程的寫操作。

  • 對MyISAM表的寫操作,會阻塞其它進程對同一表的讀和寫操作,只有當寫鎖釋放後,纔會執行其它進程的讀寫操作。

MyISAM不適合做寫爲主表的引擎,因爲寫鎖後,其它線程不能做任何操作,大量的更新會使查詢很難得到鎖,從而造成永遠阻塞

 行鎖

會出現死鎖,發生鎖衝突機率低,併發高。

在MySQL的InnoDB引擎支持行鎖,與Oracle不同,MySQL的行鎖是通過索引加載的,也就是說,行鎖是加在索引響應的行上的,要是對應的SQL語句沒有走索引,則會全表掃描,行鎖則無法實現,取而代之的是表鎖,此時其它事務無法對當前表進行更新或插入操作。

 

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

-- 這裏,我們建一個user表,主鍵爲id



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



-- A通過name執行插入操作,但事務未提交
update user set count=10 where name='xxx';
-- B在此時也執行更新操作
update user set count=10 where id=2;

-- 由於是通過非主鍵或索引選中的,升級爲爲表級鎖,
-- B則無法對該表進行更新或插入操作,只有當A提交事務後,B纔會成功執行

 for update

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

-- A用戶對id=1的記錄進行加鎖
select * from user where id=1 for update;

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

-- A用戶commit以後則B用戶可以對該記錄進行操作

行鎖的實現需要注意:

  1. 行鎖必須有索引才能實現,否則會自動鎖全表,那麼就不是行鎖了。

  2. 兩個事務不能鎖同一個索引。

  3. insert,delete,update在事務中都會自動默認加上排它鎖。

行鎖場景:

A用戶消費,service層先查詢該用戶的賬戶餘額,若餘額足夠,則進行後續的扣款操作;這種情況查詢的時候應該對該記錄進行加鎖。

否則,B用戶在A用戶查詢後消費前先一步將A用戶賬號上的錢轉走,而此時A用戶已經進行了用戶餘額是否足夠的判斷,則可能會出現餘額已經不足但卻扣款成功的情況。

爲了避免此情況,需要在A用戶操作該記錄的時候進行for update加鎖

間隙鎖

當我們用範圍條件而不是相等條件檢索數據,並請求共享或排他鎖時,InnoDB會給符合條件的已有數據記錄的索引項加鎖;對於鍵值在條件範圍內並不存在的記錄,叫做間隙

InnoDB也會對這個"間隙"加鎖,這種鎖機制就是所謂的間隙鎖

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

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

如果用戶A在進行了上述操作後,事務還未提交,則B無法對2~6之間的記錄進行更新或插入記錄,會阻塞,當A將事務提交後,B的更新操作會執行。

 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章