鎖是計算機協調多個進程或線程併發訪問某一資源的機制。html
由於數據也是一種供許多用戶共享的資源,如何保證數據併發訪問的一致性、有效性是全部數據庫必須解決的一個問題,鎖衝突也是影響數據庫併發訪問性能的一個重要因素,因此進一步學習MySQL,就須要去了解它的鎖機制。mysql
本文主要記錄學習了 MyISAM 和 InnoDB 這兩個存儲引擎,並且更加關注的是 InnoDB(由於常常用😁)git
相對其餘數據庫而言,MySQL 的鎖機制比較簡單,其最顯著的特色是不一樣的存儲引擎支持不一樣的鎖機制。好比,MyISAM和MEMORY存儲引擎採用的是表級鎖(table-level locking);BDB存儲引擎採用的是頁面鎖(page-level locking),但也支持表級鎖;InnoDB存儲引擎既支持行級鎖(row-level locking),也支持表級鎖,但默認狀況下是採用行級鎖。 MySQL這3種鎖的特性可大體概括以下。github
開銷、加鎖速度、死鎖、粒度、併發性能 ①:表級鎖:開銷小,加鎖快;不會出現死鎖;鎖定粒度大,發生鎖衝突的機率最高,併發度最低。spring
②:行級鎖:開銷大,加鎖慢;會出現死鎖;鎖定粒度最小,發生鎖衝突的機率最低,併發度也最高。sql
③:頁面鎖:開銷和加鎖時間界於表鎖和行鎖之間;會出現死鎖;鎖定粒度界於表鎖和行鎖之間,併發度通常。數據庫
從上述特色可見,很難籠統地說哪一種鎖更好,只能就具體應用的特色來講哪一種鎖更合適!僅從鎖的角度來講:表級鎖更適合於以查詢爲主,只有少許按索引條件更新數據的應用,如Web應用;而行級鎖則更適合於有大量按索引條件併發更新少許不一樣數據,同時又有併發查詢的應用,如一些在線事務處理(OLTP)系統。因爲BDB已經被InnoDB取代,即將成爲歷史(因此如今基本都在使用InnoDB存儲引擎)。session
MyISAM 存儲引擎只支持表鎖,這也是 MySQL 開始幾個版本中惟一支持的鎖類型。數據結構
mysql> show status like 'table%';
+----------------------------+-------+
| Variable_name | Value |
+----------------------------+-------+
| Table_locks_immediate | 4 |
| Table_locks_waited | 0 |
| Table_open_cache_hits | 4 |
| Table_open_cache_misses | 8 |
| Table_open_cache_overflows | 0 |
+----------------------------+-------+
5 rows in set (0.00 sec)
複製代碼
若是 Table_locks_waited 的值比較高,則說明存在着較嚴重的表級鎖爭用狀況。併發
MySQL中的表鎖兼容性:
請求鎖模式 矩陣結果表示是否兼容 當前鎖模式 |
None | 讀鎖 | 寫鎖 |
---|---|---|---|
讀鎖 | 是 | 是 | 否 |
寫鎖 | 是 | 否 | 否 |
也就是說,在MyISAM讀模式下,不會阻塞其它用戶的同一表讀操做,可是會阻塞寫操做;而在寫模式下,會同時阻塞其它用戶同一表的讀寫操做。
新建一個user表,引擎是MyISAM:
mysql> desc user;
+---------+-------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+---------+-------------+------+-----+---------+----------------+
| id | int(11) | NO | PRI | NULL | auto_increment |
| name | varchar(20) | YES | | NULL | |
| age | int(3) | YES | | NULL | |
| address | varchar(60) | YES | | NULL | |
+---------+-------------+------+-----+---------+----------------+
4 rows in set (0.01 sec)
複製代碼
session A | session B |
---|---|
得到user表的鎖鎖定 mysql> lock table user write; Query OK, 0 rows affected (0.00 sec) mysql>select * from user; Empty set (0.00 sec) mysql> insert into user(id, name, age, address) values(1, 'test', 18, 'test address'); Query OK,1 row affected (0.02 sec) |
|
mysql> select * from user\G 被阻塞了,一直卡住在這,沒有返回結果 |
|
mysql> unlock tables; Query OK, 0 rows affected (0.00 sec) |
等待 |
mysql> select * from user\G ********** name: test age: 18 address: test address 1 row in set (5 min 29.61 sec) |
能夠看出,經過lock table user write
將user表鎖住後,其它用戶進行對該表操做時,都會被阻塞。
在用LOCK TABLES給表顯式加表鎖時,必須同時取得全部涉及到表的鎖,而且MySQL不支持鎖升級。也就是說,在執行LOCK TABLES後,只能訪問顯式加鎖的這些表,不能訪問未加鎖的表;同時,若是加的是讀鎖,那麼只能執行查詢操做,而不能執行更新操做。其實,在自動加鎖的狀況下也基本如此,MyISAM老是一次得到SQL語句所須要的所有鎖。這也正是MyISAM表不會出現死鎖(Deadlock Free)的緣由。
session A | session B |
---|---|
得到user表的讀鎖定 mysql> lock table user read; Query OK, 0 rows affected (0.00 sec) |
|
mysql> select * from user where id = 1 \G 中從查詢速度中能夠看出,sessionB並無被阻塞 1 row in set (0.00 sec) |
|
因爲沒有獲取order表的讀鎖定,因此不能查詢order表 mysql> select * from order ;ERROR 1100 (HY000): Table 'order' was not locked with LOCK TABLES |
可是session B能夠訪問oder表,不阻塞 mysql> select * from order ;Empty set (0.00 sec) |
得到讀鎖定時,不能進行寫操做 mysql> update user set name = 'wahaha' where id = 1; ERROR 1099 (HY000): Table 'user' was locked with a READ lock and can't be updated |
其它session進行更新操做時,會被阻塞 mysql> update user set name = 'wahaha' where id = 1; 等待ing |
釋放鎖 mysql> unlock tables; Query OK, 0 rows affected (0.00 sec) |
等待 |
mysql> update user set name = 'wahaha' where id = 1; Query OK, 1 row affected (1 min 6.43 sec) |
MyISAM表的讀和寫是串行的,但這是就整體而言的。在必定條件下,MyISAM表也支持查詢和插入操做的併發進行。MyISAM存儲引擎有一個系統變量concurrent_insert,專門用以控制其併發插入的行爲,其值分別能夠爲0、1或2。
MyISAM存儲引擎的讀鎖和寫鎖是互斥的,讀寫操做是串行的。 但它認爲寫鎖的優先級比讀鎖高,因此即便讀請求先到鎖等待隊列,寫請求後到,寫鎖也會插到讀鎖請求以前! 這也正是MyISAM表不太適合於有大量更新操做和查詢操做應用的緣由,由於,大量的更新操做會形成查詢操做很難得到讀鎖,從而可能永遠阻塞。 能夠經過一些設置來調節MyISAM的調度行爲。
雖然上面3種方法都是要麼更新優先,要麼查詢優先的方法,但仍是能夠用其來解決查詢相對重要的應用(如用戶登陸系統)中,讀鎖等待嚴重的問題。 另外,MySQL也提供了一種折中的辦法來調節讀寫衝突,即給系統參數max_write_lock_count設置一個合適的值,當一個表的讀鎖達到這個值後,MySQL就暫時將寫請求的優先級下降,給讀進程必定得到鎖的機會。 上面已經討論了寫優先調度機制帶來的問題和解決辦法。這裏還要強調一點:一些須要長時間運行的查詢操做,也會使寫進程「餓死」!所以,應用中應儘可能避免出現長時間運行的查詢操做,不要總想用一條SELECT語句來解決問題,由於這種看似巧妙的SQL語句,每每比較複雜,執行時間較長,在可能的狀況下能夠經過使用中間表等措施對SQL語句作必定的「分解」,使每一步查詢都能在較短期完成,從而減小鎖衝突。若是複雜查詢不可避免,應儘可能安排在數據庫空閒時段執行,好比一些按期統計能夠安排在夜間執行。
InnoDB與MyISAM的最大不一樣有兩點:一是支持事務(TRANSACTION);二是採用了行級鎖。行級鎖與表級鎖原本就有許多不一樣之處,另外,事務的引入也帶來了一些新問題。
學習Spring的時候,通常經過註解@Transitional
就能啓動spring的事務管理,在MySQL中也一樣支持事務的四個原則ACID:
相對於串行處理來講,併發事務處理能大大增長數據庫資源的利用率,提升數據庫系統的事務吞吐量,從而能夠支持更多的用戶。但併發事務處理也會帶來一些問題,主要包括如下幾種狀況。
數據庫的事務隔離越嚴格,併發反作用越小,但付出的代價也就越大,由於事務隔離實質上就是使事務在必定程度上 「串行化」進行,這顯然與「併發」是矛盾的。同時,不一樣的應用對讀一致性和事務隔離程度的要求也是不一樣的,好比許多應用對「不可重複讀」和「幻讀」並不敏感,可能更關心數據併發訪問的能力。
4種隔離級別比較
讀數據一致性及容許的併發反作用 隔離級別 |
讀數據一致性 | 髒讀 | 不可重複讀 | 幻讀 |
---|---|---|---|---|
未提交讀(Read uncommitted) | 最低級別,只能保證不讀取 物理上損害的數據 |
是 | 是 | 是 |
已提交讀(Read committed) | 語句級 | 否 | 是 | 是 |
可重複讀(Repeatable read) | 事務級 | 否 | 否 | 是 |
可序列化(Serializable) | 最高級別,事務級 | 否 | 否 | 否 |
檢查InnoDB_row_lock狀態變量來分析:
mysql> show status like 'InnoDB_row_lock%';
+-------------------------------+-------+
| Variable_name | Value |
+-------------------------------+-------+
| Innodb_row_lock_current_waits | 0 |
| Innodb_row_lock_time | 0 |
| Innodb_row_lock_time_avg | 0 |
| Innodb_row_lock_time_max | 0 |
| Innodb_row_lock_waits | 0 |
+-------------------------------+-------+
5 rows in set (0.00 sec)
複製代碼
若是InnoDB_row_lock_waits和InnoDB_row_lock_time_avg的值比較高,表示鎖爭用狀況比較嚴重。
InnoDB實現了一下兩種類型的行鎖:
另外,爲了容許行鎖和表鎖共存,實現多粒度鎖機制,InnoDB還有兩種內部使用的意向鎖(Intention Locks),這兩種意向鎖都是表鎖。(感受與MyISAM的表鎖機制相似)
InnoDB行鎖模式兼容性列表:
請求鎖模式 矩陣結果表示是否兼容 當前鎖模式 |
X | IX | S | IS |
---|---|---|---|---|
X | 衝突 | 衝突 | 衝突 | 衝突 |
IX | 衝突 | 兼容 | 衝突 | 兼容 |
S | 衝突 | 衝突 | 兼容 | 兼容 |
IS | 衝突 | 兼容 | 兼容 | 兼容 |
若是一個事務請求的鎖模式與當前的鎖兼容,InnoDB就將請求的鎖授予該事務;反之,若是二者不兼容,該事務就要等待鎖釋放。 意向鎖是InnoDB自動加的;對於UPDATE、DELETE和INSERT語句,InnoDB會自動給設計數據集加排他鎖(X);對於普通的SELECT語句,InnoDB不會加鎖。 能夠經過如下語句顯示給記錄集加共享鎖或排他鎖:
用SELECT ... IN SHARE MODE得到共享鎖,主要用在須要數據依存關係時來確認某行記錄是否存在,並確保沒有人對這個記錄進行UPDATE或者DELETE操做。可是若是當前事務也須要對該記錄進行更新操做,則頗有可能形成死鎖,對於鎖定行記錄後須要進行更新操做的應用,應該使用SELECT... FOR UPDATE方式得到排他鎖。
因此在使用共享鎖模式下,查詢完數據後不要進行更新操做,否則又可能會形成死鎖;要更新數據,應該使用排他鎖模式。
InnoDB行鎖是經過給索引上的索引項加鎖來實現的,這一點MySQL與Oracle不一樣,後者是經過在數據塊中對相應數據行加鎖來實現的。InnoDB這種行鎖實現特色意味着:只有經過索引條件檢索數據,InnoDB才使用行級鎖,不然,InnoDB將使用表鎖!(這個問題遇到過,因爲沒加索引,行鎖變表鎖)
能夠經過explain執行計劃查看是否真正使用了索引。
當咱們用範圍條件而不是相等條件檢索數據,並請求共享或排他鎖時,InnoDB會給符合條件的已有數據記錄的索引項加鎖;對於鍵值在條件範圍內但並不存在的記錄,叫作「間隙(GAP)」,InnoDB也會對這個「間隙」加鎖,這種鎖機制就是所謂的間隙鎖(Next-Key鎖)。
舉個🌰:
假如emp表中只有101條記錄,其id的值從1~101,下面的sql: select * from emp where id > 100 for update; 是範圍條件查詢,InnoDB不只會對符合條件的id值爲101的記錄加鎖,也會對id大於101(並不存在的值)的「間隙」加鎖。
結論:
很顯然,在使用範圍條件檢索並鎖定記錄時,InnoDB這種加鎖機制會阻塞符合條件範圍內鍵值的併發插入,這每每會形成嚴重的鎖等待。所以,在實際應用開發中,尤爲是併發插入比較多的應用,咱們要儘可能優化業務邏輯,儘可能使用相等條件來訪問更新數據,避免使用範圍條件。
上面知識點說過,MyISAM表鎖是deadlock free的,這是由於MyISAM老是一次得到所需的所有鎖,要麼所有知足,要麼等待,所以不會出現死鎖。但在InnoDB中,除單個SQL組成的事務外,鎖是逐步或得的,因此InnoDB發生死鎖是可能的。
舉個🌰:
session A | session B |
---|---|
mysql> set autocommit = 0; Query OK, 0 rows affected (0.00 sec) mysql> select * from table_1 where where id=1 for update; ... 作一些其餘處理... |
mysql> set autocommit = 0; Query OK, 0 rows affected (0.00 sec) mysql> select * from table_2 where id=1 for update; ... |
select * from table_2 where id =1 for update; 因session_2已取得排他鎖,等待 |
作一些其餘處理... |
mysql> select * from table_1 where where id=1 for update; 死鎖 |
也就是咱們死鎖產生的條件,互相持有資源不釋放,還有環形等待。
發生死鎖後,InnoDB通常都能自動檢測到,並使一個事務釋放鎖並回退,另外一個事務得到鎖,繼續完成事務。但在涉及外部鎖,或涉及表鎖的狀況下,InnoDB並不能徹底自動檢測到死鎖,這須要經過設置鎖等待超時參數 innodb_lock_wait_timeout來解決。須要說明的是,這個參數並非只用來解決死鎖問題,在併發訪問比較高的狀況下,若是大量事務因沒法當即得到所需的鎖而掛起,會佔用大量計算機資源,形成嚴重性能問題,甚至拖跨數據庫。咱們經過設置合適的鎖等待超時閾值,能夠避免這種狀況發生。
這是一篇學習文章,關於MySQL的鎖機制又多了幾分瞭解,之後在寫SQL和排查問題時候,儘可能避免死鎖和更快定位問題所在。