mysql中Innodb 行鎖實(shí)現(xiàn)原理
一、Innodb行鎖的實(shí)現(xiàn)
【1】Innodb的行鎖是通過給索引的索引項(xiàng)加鎖來實(shí)現(xiàn)的
【2】Innodb按照輔助索引進(jìn)行數(shù)據(jù)操作時(shí),輔助索引和主鍵索引都將鎖定指定的索引項(xiàng)
【3】通過索引進(jìn)行數(shù)據(jù)檢索時(shí),Innodb才使用行級鎖,否則Innodb將使用表鎖
二、場景分析
環(huán)境: 創(chuàng)建一張表,ID為主鍵,Name為普通字段。下面通過實(shí)際場景進(jìn)行說明。
【1】使用主鍵ID來進(jìn)行數(shù)據(jù)檢索時(shí),其余行仍然可以操作。
| session1 | session2 |
|---|---|
| mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) | mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) |
| mysql> select * from test_lock wehre id = 1 for update; ±-----±-----+ | id | name | | 1 | hua zi | 1 row in set(0.00sec) | |
| mysql> select * from test_lock wehre id = 2 for update; ±-----±-----+ | id | name | | 2 | guo zi | 1 row in set(0.00sec) | |
| 【2】用非索引的字段來進(jìn)行數(shù)據(jù)檢索時(shí),此時(shí)會(huì)升級為表鎖,其余列就不能操作。 | |
| session1 | session2 |
| ---- | ---- |
| mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) | mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) |
| mysql> select * from test_lock wehre name = ‘hua zi’ for update; ±-----±-----+ | id | name | | 1 | hua zi | 1 row in set(0.00sec) | |
| mysql> select * from test_lock wehre name = ‘guo zi’ for update; 等待 | |
【3】由于MySQL的行鎖是針對索引加的鎖,不是針對記錄加的鎖,所以雖然是訪問不同行的記錄,但是如果是使用相同的索引鍵,是會(huì)出現(xiàn)鎖沖突的。應(yīng)用設(shè)計(jì)的時(shí)候要注意這一點(diǎn)。雖然 session_2和session_1訪問的是不同的記錄,但因?yàn)槭褂昧讼嗤乃饕孕枰却i。 | |
| session1 | session2 |
| ---- | ---- |
| mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) | mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) |
| mysql> select * from test_lock wehre id = 1 and name = ‘hua zi’ for update; ±-----±-----+ | id | name | | 1 | hua zi | 1 row in set(0.00sec) | |
| mysql> select * from test_lock wehre id = 1 and name = ‘guo zi’ for update; 等待 | |
【4】當(dāng)表有多個(gè)索引的時(shí)候,不同的事務(wù)可以使用不同的索引鎖定不同的行,另外,不論是使用主鍵索引、唯一索引或普通索引,InnoDB都會(huì)使用行鎖來對數(shù)據(jù)加鎖。 | |
| session1 | session2 |
| ---- | ---- |
| mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) | mysql>set autocommit=0; Query OK,0 rows affected(0.00sec) |
| mysql> select * from test_lock wehre id = 1 for update; ±-----±-----+ | id | name | | 1 | hua zi | 1 row in set(0.00sec) | |
| mysql> select * from test_lock wehre name = ‘guo zi’ for update; ±-----±-----+ | id | name | | 2 | guo zi | 1 row in set(0.00sec) | |
| mysql> select * from test_lock wehre name = ‘hua zi’ for update; 等待 |
三、特殊場景
即便在條件中使用了索引字段,但是否使用索引來檢索數(shù)據(jù)是由MySQL通過判斷不同執(zhí)行計(jì)劃的代價(jià)來決定的,如果MySQL認(rèn)為全表掃描效率更高,比如對一些很小的表,它就不會(huì)使用索引,這種情況下InnoDB將使用表鎖,而不是行鎖。因此,在分析鎖沖突時(shí),別忘了檢查SQL的執(zhí)行計(jì)劃,以確認(rèn)是否真正使用了索引。
在下面的例子中,檢索值的數(shù)據(jù)類型與索引字段不同,雖然MySQL能夠進(jìn)行數(shù)據(jù)類型轉(zhuǎn)換,但卻不會(huì)使用索引,從而導(dǎo)致InnoDB使用表鎖。通過用explain檢查兩條SQL的執(zhí)行計(jì)劃,我們可以清楚地看到了這一點(diǎn)。
mysql> alter table test_lock add index name(name); Query OK, 4 rows affected (8.06 sec) Records: 4 Duplicates: 0 Warnings: 0 mysql> explain select * from test_lock where name = 1 \G *************************** 1. row *************************** id: 1 select_type: SIMPLE table: test_lock type: ALL possible_keys: name key: NULL key_len: NULL ref: NULL rows: 4 Extra: Using where 1 row in set (0.00 sec) mysql> explain select * from test_lock where name = '1' \G *************************** 1. row *************************** id: 1 select_type: SIMPLE table: test_lock type: ref possible_keys: name key: name key_len: 23 ref: const rows: 1 Extra: Using where 1 row in set (0.00 sec)
到此這篇關(guān)于mysql中Innodb 行鎖實(shí)現(xiàn)原理的文章就介紹到這了,更多相關(guān)mysql Innodb 行鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL數(shù)據(jù)表損壞的正確修復(fù)方案
修復(fù)以損壞的MySQL數(shù)據(jù)表的實(shí)際操作在實(shí)際中是我們經(jīng)常用到的,以下的文章主要是介紹正確修復(fù)以損壞的MySQL數(shù)據(jù)表的實(shí)際操作步驟,以下就是正文的介紹,希望會(huì)給你帶來一些幫助在此方面。2011-01-01
LEFT JOIN關(guān)聯(lián)表中ON,WHERE后面跟條件的區(qū)別
本文主要介紹了LEFT JOIN關(guān)聯(lián)表中ON,WHERE后面跟條件的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01
MySQL MHA 運(yùn)行狀態(tài)監(jiān)控介紹
這篇文章主要介紹MySQL MHA 運(yùn)行狀態(tài)監(jiān)控,MHA(Master HA)是一款開源的 MySQL 的高可用程序,它為 MySQL 主從復(fù)制架構(gòu)提供了 automating master failover 功能,想具體了解的小伙伴可以和小編一起學(xué)習(xí)下面文章內(nèi)容2021-10-10
mysql運(yùn)行net start mysql報(bào)服務(wù)名無效的解決辦法
這篇文章主要為大家詳細(xì)介紹了mysql運(yùn)行net start mysql報(bào)服務(wù)名無效的解決辦法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01

