Mysql中的行级锁、表级锁、页级锁
轉(zhuǎn)載自?Mysql中的行級(jí)鎖、表級(jí)鎖、頁級(jí)鎖
?
在計(jì)算機(jī)科學(xué)中,鎖是在執(zhí)行多線程時(shí)用于強(qiáng)行限制資源訪問的同步機(jī)制,即用于在并發(fā)控制中保證對(duì)互斥要求的滿足。
在數(shù)據(jù)庫的鎖機(jī)制中介紹過,在DBMS中,可以按照鎖的粒度把數(shù)據(jù)庫鎖分為行級(jí)鎖(INNODB引擎)、表級(jí)鎖(MYISAM引擎)和頁級(jí)鎖(BDB引擎 )。
行級(jí)鎖
行級(jí)鎖是Mysql中鎖定粒度最細(xì)的一種鎖,表示只針對(duì)當(dāng)前操作的行進(jìn)行加鎖。行級(jí)鎖能大大減少數(shù)據(jù)庫操作的沖突。其加鎖粒度最小,但加鎖的開銷也最大。行級(jí)鎖分為共享鎖?和?排他鎖。
特點(diǎn)
開銷大,加鎖慢;會(huì)出現(xiàn)死鎖;鎖定粒度最小,發(fā)生鎖沖突的概率最低,并發(fā)度也最高。
表級(jí)鎖
表級(jí)鎖是MySQL中鎖定粒度最大的一種鎖,表示對(duì)當(dāng)前操作的整張表加鎖,它實(shí)現(xiàn)簡(jiǎn)單,資源消耗較少,被大部分MySQL引擎支持。最常使用的MYISAM與INNODB都支持表級(jí)鎖定。表級(jí)鎖定分為表共享讀鎖(共享鎖)與表獨(dú)占寫鎖(排他鎖)。
特點(diǎn)
開銷小,加鎖快;不會(huì)出現(xiàn)死鎖;鎖定粒度大,發(fā)出鎖沖突的概率最高,并發(fā)度最低。
頁級(jí)鎖
表級(jí)鎖是MySQL中鎖定粒度介于行級(jí)鎖和表級(jí)鎖中間的一種鎖.表級(jí)鎖速度快,但沖突多,行級(jí)沖突少,但速度慢。所以取了折衷的頁級(jí),一次鎖定相鄰的一組記錄。BDB支持頁級(jí)鎖
特點(diǎn)
開銷和加鎖時(shí)間界于表鎖和行鎖之間;會(huì)出現(xiàn)死鎖;鎖定粒度界于表鎖和行鎖之間,并發(fā)度一般
MySQL常用存儲(chǔ)引擎的鎖機(jī)制
MyISAM和MEMORY采用表級(jí)鎖(table-level locking)
BDB采用頁面鎖(page-level locking)或表級(jí)鎖,默認(rèn)為頁面鎖
InnoDB支持行級(jí)鎖(row-level locking)和表級(jí)鎖,默認(rèn)為行級(jí)鎖
Innodb中的行鎖與表鎖
前面提到過,在Innodb引擎中既支持行鎖也支持表鎖,那么什么時(shí)候會(huì)鎖住整張表,什么時(shí)候或只鎖住一行呢?
InnoDB行鎖是通過給索引上的索引項(xiàng)加鎖來實(shí)現(xiàn)的,這一點(diǎn)MySQL與Oracle不同,后者是通過在數(shù)據(jù)塊中對(duì)相應(yīng)數(shù)據(jù)行加鎖來實(shí)現(xiàn)的。InnoDB這種行鎖實(shí)現(xiàn)特點(diǎn)意味著:只有通過索引條件檢索數(shù)據(jù),InnoDB才使用行級(jí)鎖,否則,InnoDB將使用表鎖!
在實(shí)際應(yīng)用中,要特別注意InnoDB行鎖的這一特性,不然的話,可能導(dǎo)致大量的鎖沖突,從而影響并發(fā)性能。
行級(jí)鎖都是基于索引的,如果一條SQL語句用不到索引是不會(huì)使用行級(jí)鎖的,會(huì)使用表級(jí)鎖。行級(jí)鎖的缺點(diǎn)是:由于需要請(qǐng)求大量的鎖資源,所以速度慢,內(nèi)存消耗大。
行級(jí)鎖與死鎖
MyISAM中是不會(huì)產(chǎn)生死鎖的,因?yàn)镸yISAM總是一次性獲得所需的全部鎖,要么全部滿足,要么全部等待。而在InnoDB中,鎖是逐步獲得的,就造成了死鎖的可能。
在MySQL中,行級(jí)鎖并不是直接鎖記錄,而是鎖索引。索引分為主鍵索引和非主鍵索引兩種,如果一條sql語句操作了主鍵索引,MySQL就會(huì)鎖定這條主鍵索引;如果一條語句操作了非主鍵索引,MySQL會(huì)先鎖定該非主鍵索引,再鎖定相關(guān)的主鍵索引。 在UPDATE、DELETE操作時(shí),MySQL不僅鎖定WHERE條件掃描過的所有索引記錄,而且會(huì)鎖定相鄰的鍵值,即所謂的next-key locking。
當(dāng)兩個(gè)事務(wù)同時(shí)執(zhí)行,一個(gè)鎖住了逐漸索引在等待其他相關(guān)索引,一個(gè)鎖定了非主鍵索引,在等待主鍵索引。這樣就會(huì)發(fā)生死鎖。
發(fā)生死鎖后,InnoDB一般都可以檢測(cè)到,并使一個(gè)事務(wù)釋放鎖回退,另一個(gè)獲取鎖完成事務(wù)。
有多種方法可以避免死鎖,這里只介紹常見的三種
1、如果不同程序會(huì)并發(fā)存取多個(gè)表,盡量約定以相同的順序訪問表,可以大大降低死鎖機(jī)會(huì)。
2、在同一個(gè)事務(wù)中,盡可能做到一次鎖定所需要的所有資源,減少死鎖產(chǎn)生概率;
3、對(duì)于非常容易產(chǎn)生死鎖的業(yè)務(wù)部分,可以嘗試使用升級(jí)鎖定顆粒度,通過表級(jí)鎖定來減少死鎖產(chǎn)生的概率;
?
總結(jié)
以上是生活随笔為你收集整理的Mysql中的行级锁、表级锁、页级锁的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 2000配置电脑主机清单(2000配置电
- 下一篇: Java开发2018年值得学习的10大技