InnoDB支持事務,MyISAM不支持事務.mysql
一.事務的基本特性sql
ACID特性數據庫
1.原子性(Atomicity):事務是一個原子操做單元,其對數據的修改,要麼全都執行,要麼全都不執行。數據結構
2.一致性(Consistent):在事務開始和完成時,數據都必須保持一致狀態。這意味着全部相關的數據規則都必須應用於事務的修改,以保持數據的完整性;事務結束時,全部的內部數據結構(如B樹索引或雙向鏈表)也都必須是正確的。併發
3. 隔離性(Isolation):數據庫系統提供必定的隔離機制,保證事務在不受外部併發操做影響的「獨立」環境執行。這意味着事務處理過程當中的中間狀態對外部是不可見的,反之亦然。性能
4.持久性(Durable):事務完成以後,它對於數據的修改是永久性的,即便出現系統故障也可以保持。優化
二.併發事務帶來的問題spa
1.更新丟失線程
假設table中的price在更新前爲03d
如A用戶開始一個事務:
BEGIN;
SELECT price FROM table WHERE id=1;
#開始更新
UPDATE table SET price = price + 1 WHERE id=1;
COMMIT;
B用戶在A用戶未提交事務時,一樣更新:
BEGIN;
SELECT price FROM table WHERE id=1; #此處的price應該是1,可是A用戶未提交事務,因此仍是0
#開始更新
UPDATE table SET price = price+2 WHERE id=1;
COMMIT;
最終price=2,但實際上應該是3,這就是更新丟失
2.髒讀
一個事務正在對一條記錄作修改,在這個事務完成並提交前,這條記錄的數據就處於不一致狀態;這時,另外一個事務也來讀取同一條記錄,若是不加控制,第二個事務讀取了這些「髒」數據,並據此作進一步的處理,就會產生未提交的數據依賴關係。這種現象被形象地叫作"髒讀".上面B用戶在讀取price時,A用戶未提交事務,B用戶讀到就是髒數據.
3.不可重複讀
一個事務在讀取某些數據後的某個時間,再次讀取之前讀過的數據,卻發現其讀出的數據已經發生了改變、或某些記錄已經被刪除了!這種現象就叫作「不可重複讀」。
4.幻讀
一個事務按相同的查詢條件從新讀取之前檢索過的數據,卻發現其餘事務插入了知足其查詢條件的新數據,這種現象就稱爲「幻讀」。
三.事務的隔離級別
在上面講到的併發事務處理帶來的問題中,「更新丟失」一般是應該徹底避免的。但防止更新丟失,並不能單靠數據庫事務控制器來解決,須要應用程序對要更新的數據加必要的鎖來解決,所以,防止更新丟失應該是應用的責任。
「髒讀」、「不可重複讀」和「幻讀」,其實都是數據庫讀一致性問題,必須由數據庫提供必定的事務隔離機制來解決。數據庫實現事務隔離的方式,基本上可分爲如下兩種。
1.一種是在讀取數據前,對其加鎖,阻止其餘事務對數據進行修改。
2.另外一種是不用加任何鎖,經過必定機制生成一個數據請求時間點的一致性數據快照(Snapshot),並用這個快照來提供必定級別(語句級或事務級)的一致性讀取。從用戶的角度來看,好象是數據庫能夠提供同一數據的多個版本,所以,這種技術叫作數據多版本併發控制(MultiVersion Concurrency Control,簡稱MVCC或MCC),也常常稱爲多版本數據庫。
3.mysql的四種隔離級別
Ps:
a.Repeatable read是默認隔離級別.
b. 查看當前會話隔離級別:select @@tx_isolation;
c.查看系統隔離級別: select @@global.tx_isolation;
四.InnoDB鎖爭用:
mysql> show global status like 'innodb_row_lock%';
Innodb_row_lock_current_waits 0
Innodb_row_lock_time 530
Innodb_row_lock_time_avg 106
Innodb_row_lock_time_max 327
Innodb_row_lock_waits 5
若是發現鎖爭用比較嚴重,如InnoDB_row_lock_waits和InnoDB_row_lock_time_avg的值比較高,還能夠經過設置InnoDB Monitors來進一步觀察發生鎖衝突的表、數據行等,並分析鎖爭用的緣由。
Ps:InnoDB鎖超時時間由變量innodb_lock_wait_timeout控制,默認是50s
五.InnoDB的行鎖模式及加鎖方法
(一).行鎖模式
1.共享鎖(S鎖):
對同一行數據均可以共享一把鎖,可是沒有得到鎖的事務只能夠讀,不能夠修改
2.排它鎖(X鎖)
對同一行數據,得到該鎖的事務可讀可寫,未得到鎖的事務不可讀也不可寫.
另外,爲了容許行鎖和表鎖共存,實現多粒度鎖機制,InnoDB還有兩種內部使用的意向鎖(Intention Locks),這兩種意向鎖都是表鎖。
3.意向共享鎖(IS):事務打算給數據行加行共享鎖,事務在給一個數據行加共享鎖前必須先取得該表的IS鎖。
4.意向排他鎖(IX):事務打算給數據行加行排他鎖,事務在給一個數據行加排他鎖前必須先取得該表的IX鎖。
若是一個事務請求的鎖模式與當前的鎖兼容,InnoDB就將請求的鎖授予該事務;反之,若是二者不兼容,該事務就要等待鎖釋放。
意向鎖是InnoDB自動加的,不需用戶干預。對於UPDATE、DELETE和INSERT語句,InnoDB會自動給涉及數據集加排他鎖(X);對於普通SELECT語句,InnoDB不會加任何鎖;事務能夠經過如下語句顯示給記錄集加共享鎖或排他鎖。
(二).加鎖方法:
Ps:
1. 共享鎖
SELECT ... LOCK IN SHARE MODE
主要用在須要數據依存關係時來確認某行記錄是否存在,並確保沒有人對這個記錄進行UPDATE或者DELETE操做. 可是若是當前事務也須要對該記錄進行更新操做,則頗有可能形成死鎖,對於鎖定行記錄後須要進行更新操做的應用,應該使用SELECT... FOR UPDATE方式得到排他鎖。
對於加了共享鎖的數據行,其餘事務能夠加共享鎖或不加鎖,但沒法加排它鎖.
2. 排它鎖
SELECT … FOR UPDATE
六.InnoDB的行鎖實現方式
InnoDB行鎖是經過給索引上的索引項加鎖來實現的, InnoDB這種行鎖實現特色意味着:只有經過索引條件檢索數據,InnoDB才使用行級鎖,不然,InnoDB將使用表鎖!
在實際應用中,要特別注意InnoDB行鎖的這一特性,否則的話,可能致使大量的鎖衝突,從而影響併發性能。
1.在不經過索引條件查詢的時候,InnoDB使用的是表鎖,而不是行鎖
在name列上加鎖後,上述狀況將不存在
2. MySQL的行鎖是針對索引加的鎖,不是針對記錄加的鎖,因此雖然是訪問不一樣行的記錄,可是若是是使用相同的索引鍵,是會出現鎖等待的
3. 當表有多個索引的時候,不一樣的事務可使用不一樣的索引鎖定不一樣的行,另外,不管是使用主鍵索引、惟一索引或普通索引,InnoDB都會使用行鎖來對數據加鎖。
4.即使在條件中使用了索引字段,可是否使用索引來檢索數據是由MySQL經過判斷不一樣執行計劃的代價來決定的,若是MySQL認爲全表掃描效率更高,好比對一些很小的表,它就不會使用索引,這種狀況下InnoDB將使用表鎖,而不是行鎖。所以,在分析鎖衝突時,別忘了檢查SQL的執行計劃,以確認是否真正使用了索引。
以下示例(在mysql5.1版本上,5.5以上版本不存在下列問題):
name列有索引,name字段類型爲varchar.
EXPLAIN SELECT * FROM `test` WHERE `name` =1;#執行全表掃描
EXPLAIN SELECT * FROM `test` WHERE `name`='1';#執行索引掃描
七.間隙鎖
當咱們用範圍條件而不是相等條件檢索數據,並請求共享或排他鎖時,InnoDB會給符合條件的已有數據記錄的索引項加鎖;對於鍵值在條件範圍內但並不存在的記錄,叫作「間隙(GAP)」,InnoDB也會對這個「間隙」加鎖,這種鎖機制就是所謂的間隙鎖(Next-Key鎖)。
舉例來講,假如emp表中只有101條記錄,其empid的值分別是 1,2,...,100,101,下面的SQL:
Select * from emp where empid > 100 for update;
是一個範圍條件的檢索,InnoDB不只會對符合條件的empid值爲101的記錄加鎖,也會對empid大於101(這些記錄並不存在)的「間隙」加鎖。
InnoDB使用間隙鎖的目的,一方面是爲了防止幻讀,以知足相關隔離級別的要求,對於上面的例子,要是不使用間隙鎖,若是其餘事務插入了empid大於100的任何記錄,那麼本事務若是再次執行上述語句,就會發生幻讀;另一方面,是爲了知足其恢復和複製的須要。有關其恢復和複製對鎖機制的影響,以及不一樣隔離級別下InnoDB使用間隙鎖的狀況,在後續的會作進一步介紹。
很顯然,在使用範圍條件檢索並鎖定記錄時,InnoDB這種加鎖機制會阻塞符合條件範圍內鍵值的併發插入,這每每會形成嚴重的鎖等待。所以,在實際應用開發中,尤爲是併發插入比較多的應用,咱們要儘可能優化業務邏輯,儘可能使用相等條件來訪問更新數據,避免使用範圍條件。
還要特別說明的是,InnoDB除了經過範圍條件加鎖時使用間隙鎖外,若是使用相等條件請求給一個不存在的記錄加鎖,InnoDB也會使用間隙鎖!
PS:
在多列條件查找時,
如SELECT * FROM table WHERE col1=2 AND col2>1000 FOR UPDATE;在col1和col2上均有索引
這個時候
INSERT INTO table(col1,col2) VALUES(3,8);能夠執行
INSERT INTO table(col1,col2) VALUES(2,8);不能夠執行
InnoDB根據索引只是鎖定了須要鎖定的間隙鎖.
儘可能少用不肯定的SQL語句如
insert into target_tab select * from source_tab where ...
create table new_tab ...select ... From source_tab where ...(CTAS)
經過使用「select * from source_tab ... Into outfile」和「load data infile ...」語句組合來間接實現,採用這種方式MySQL不會給source_tab加鎖
七.InnoDB的表鎖
注意點:
1.LOCK TABLES tb_name WRITE;//當前會話對錶tb_name可讀可寫,其他會話對錶tb_name不可讀不可寫
2.LOCK TABLES tb_name READ;//當前會話對錶tb_name可讀不可寫,其他會話對錶tb_name可讀不可寫
3.你須要一次鎖定更新的表
LOCK TABLES tb1_name WRITE;
在鎖定過程當中,你能夠讀tbl2_name,當你須要更新tbl2_name,你將獲得一個表沒法鎖定的錯誤
4.innodb的表鎖,開始事務時會自動釋放表鎖,因此begin;或set autocommit=0;等命令應該在lock tables的前面.
在InnoDB下,使用表鎖要注意如下兩點。
1.使用LOCK TABLES雖然能夠給InnoDB加表級鎖,但必須說明的是,表鎖不是由InnoDB存儲引擎層管理的,而是由其上一層──MySQL Server負責的,僅當autocommit=0、innodb_table_locks=1(默認設置)時,InnoDB層才能知道MySQL加的表鎖,MySQL Server也才能感知InnoDB加的行鎖,這種狀況下,InnoDB才能自動識別涉及表級鎖的死鎖;不然,InnoDB將沒法自動檢測並處理這種死鎖。有關死鎖,下一小節還會繼續討論。
2.在用LOCK TABLES對InnoDB表加鎖時要注意,要將AUTOCOMMIT設爲0,不然MySQL不會給表加鎖;事務結束前,不要用UNLOCK TABLES釋放表鎖,由於UNLOCK TABLES會隱含地提交事務;COMMIT或ROLLBACK並不能釋放用LOCK TABLES加的表級鎖,必須用UNLOCK TABLES釋放表鎖。正確的方式見以下語句:
例如,若是須要寫表t1並從表t讀,能夠按以下作:
SET AUTOCOMMIT=0;
LOCK TABLES t1 WRITE, t2 READ, ...;
[do something with tables t1 and t2 here];
COMMIT;
UNLOCK TABLES;
八.在應用中避免死鎖的方法
發生死鎖後,InnoDB通常都能自動檢測到,並使一個事務釋放鎖並回退,另外一個事務得到鎖,繼續完成事務。但在涉及外部鎖,或涉及表鎖的狀況下,InnoDB並不能徹底自動檢測到死鎖,這須要經過設置鎖等待超時參數innodb_lock_wait_timeout來解決。須要說明的是,這個參數並非只用來解決死鎖問題,在併發訪問比較高的狀況下,若是大量事務因沒法當即得到所需的鎖而掛起,會佔用大量計算機資源,形成嚴重性能問題,甚至拖跨數據庫。咱們經過設置合適的鎖等待超時閾值,能夠避免這種狀況發生。
1. 在應用中,若是不一樣的程序會併發存取多個表,應儘可能約定以相同的順序來訪問表,這樣能夠大大下降產生死鎖的機會。
2. 在程序以批量方式處理數據的時候,若是事先對數據排序,保證每一個線程按固定的順序來處理記錄,也能夠大大下降出現死鎖的可能。
3. 在事務中,若是要更新記錄,應該直接申請足夠級別的鎖,即排他鎖,而不該先申請共享鎖,更新時再申請排他鎖,由於當用戶申請排他鎖時,其餘事務可能又已經得到了相同記錄的共享鎖,從而形成鎖衝突,甚至死鎖。
4. 前面講過,在REPEATABLE-READ隔離級別下,若是兩個線程同時對相同條件記錄用SELECT...FOR UPDATE加排他鎖,在沒有符合該條件記錄狀況下,兩個線程都會加鎖成功。程序發現記錄尚不存在,就試圖插入一條新記錄,若是兩個線程都這麼作,就會出現死鎖。這種狀況下,將隔離級別改爲READ COMMITTED,就可避免問題
5. 當隔離級別爲READ COMMITTED時,若是兩個線程都先執行SELECT...FOR UPDATE,判斷是否存在符合條件的記錄,若是沒有,就插入記錄。此時,只有一個線程能插入成功,另外一個線程會出現鎖等待,當第1個線程提交後,第2個線程會因主鍵重複出錯,但雖然這個線程出錯了,卻會得到一個排他鎖!這時若是有第3個線程又來申請排他鎖,也會出現死鎖。對於這種狀況,能夠直接作插入操做,而後再捕獲主鍵重複異常,或者在遇到主鍵重錯誤時,老是執行ROLLBACK釋放得到的排他鎖