《Mysql學(xué)習MySQL中InnoDB存儲引擎的鎖的基本使用教程》要點:
本文介紹了Mysql學(xué)習MySQL中InnoDB存儲引擎的鎖的基本使用教程,希望對您有用。如果有疑問,可以聯(lián)系我們。
MYSQL數(shù)據(jù)庫MyISAM和MEMORY采用表級鎖(table-level locking)
MYSQL數(shù)據(jù)庫BDB采用頁面鎖(page-leve locking)或表級鎖,默認為頁面鎖
MYSQL數(shù)據(jù)庫InnoDB支持行級鎖(row-level locking)和表級鎖,默認為行級鎖
MYSQL數(shù)據(jù)庫各種鎖特點
MYSQL數(shù)據(jù)庫表級鎖:開銷小,加鎖快;不會出現(xiàn)死鎖;鎖定粒度大,發(fā)生沖突的概率最高,并發(fā)度最低
MYSQL數(shù)據(jù)庫行級鎖:開銷大,加鎖慢;會出現(xiàn)死鎖;鎖定粒度最小,發(fā)生鎖沖突的概率最低,并發(fā)度也最高
MYSQL數(shù)據(jù)庫頁面鎖:開銷和加鎖時間介于表鎖和行鎖之間;會出現(xiàn)死鎖;鎖定粒度介于表鎖和行鎖之間,并發(fā)度一般
MYSQL數(shù)據(jù)庫InnoDB存儲引擎的鎖
MYSQL數(shù)據(jù)庫InnoDB存儲引擎實現(xiàn)了如下兩種鎖
MYSQL數(shù)據(jù)庫1、共享鎖(S Lock),允許事務(wù)讀一行數(shù)據(jù)
MYSQL數(shù)據(jù)庫2、排他鎖(X Lock),允許事務(wù)更新或者刪除一行數(shù)據(jù)
MYSQL數(shù)據(jù)庫共享鎖和排他鎖的兼容如下圖所示
MYSQL數(shù)據(jù)庫一致性的非鎖定讀
MYSQL數(shù)據(jù)庫一致性的非鎖定行讀(consistent nonlocking read)是指InnoDB存儲引擎通過行多版本控制(multi versioning)的方式來讀取當前執(zhí)行時間數(shù)據(jù)庫中行的數(shù)據(jù).如果讀取的行正在執(zhí)行DELETE、UPDATE操作,這是讀取操作不會因此而會等待行上鎖的釋放,相反,InnoDB會去讀取行的一個快照數(shù)據(jù).
MYSQL數(shù)據(jù)庫之所以稱其為非鎖定讀,因為不需要等待拜訪的行上X鎖的釋放.快照數(shù)據(jù)是指改行之前版本的數(shù)據(jù),該實現(xiàn)是通過undo段來實現(xiàn)的.但是在不同事務(wù)隔離級別下,讀取的方式不同,并不是每個事務(wù)隔離級別下讀取的都是一致性讀.
MYSQL數(shù)據(jù)庫例如:
MYSQL數(shù)據(jù)庫對于read committed的事務(wù)隔離級別,他總是讀取行的最新版本,如果行被鎖定了,則讀取該行版本的最新一個快照.
MYSQL數(shù)據(jù)庫對于repeatable read(innoDB存儲引擎的默認隔離級別),總是讀取事務(wù)開始時的行數(shù)據(jù).
MYSQL數(shù)據(jù)庫?非鎖定讀的機制大大提高了數(shù)據(jù)讀取的并發(fā)性,在Innodb存儲引擎默認設(shè)置下,這是默認的讀取方式,但是在某些情況下,可以對讀進行加鎖,比如:
MYSQL數(shù)據(jù)庫1、顯式對讀進行加鎖,如使用 select --- for update ;select --- lock in share mode
MYSQL數(shù)據(jù)庫2、在外鍵的插入和更新上,因為在外鍵的插入和更新上,對于數(shù)據(jù)的隔離性要求較高,在插入前需要掃描父表中的記錄是否存在,所以,在外鍵的插入刪除上,InnoDB會使用加S鎖的方式來實現(xiàn).
MYSQL數(shù)據(jù)庫InnoDB鎖的算法
MYSQL數(shù)據(jù)庫1、Record Lock:單個行記錄上的鎖
MYSQL數(shù)據(jù)庫2、Gap Lock:間隙鎖,鎖定一個范圍,但不包括記錄本身
MYSQL數(shù)據(jù)庫3、Next-key Lock:Gap Lock+Record Lock,鎖定一個范圍,并且鎖定記錄自己
MYSQL數(shù)據(jù)庫Record Lock總是會去鎖住索引記錄,如果InnoDB存儲引擎表建立的時候沒有設(shè)置任何一個索引,這時InnodB存儲引擎會使用隱式的主鍵來進行鎖定,在Repeatable Read隔離級別下,Next-key Lock 算法是默認的行記錄鎖定算法.
MYSQL數(shù)據(jù)庫鎖帶來的問題
MYSQL數(shù)據(jù)庫1、丟失更新
MYSQL數(shù)據(jù)庫如何避免丟失更新:讓事務(wù)變成串行操作,而不是并發(fā)的操作,即對每個事務(wù)開始---對讀取記錄加排他鎖.
MYSQL數(shù)據(jù)庫2、臟讀
MYSQL數(shù)據(jù)庫臟讀即一個事務(wù)可以讀到另一個事務(wù)中未提交的數(shù)據(jù),這違反了數(shù)據(jù)庫的隔離性.
MYSQL數(shù)據(jù)庫臟讀發(fā)生的條件是需要事務(wù)的隔離級別為Read uncommitted.
MYSQL數(shù)據(jù)庫3、不可重復(fù)讀
MYSQL數(shù)據(jù)庫不可重復(fù)讀與臟讀的區(qū)別是:臟讀是讀到未提交的數(shù)據(jù),而不可重復(fù)讀讀到的是已經(jīng)提交的數(shù)據(jù).
MYSQL數(shù)據(jù)庫一般來說,不可重復(fù)讀是可以接受的,在InnoDB存儲引擎中,通過使用Next-Key Lock算法來避免不可重復(fù)讀的問題.
MYSQL數(shù)據(jù)庫值得注意的是,默認情況下InnoDB存儲引擎不會回滾超時引發(fā)的錯誤異常.
MYSQL數(shù)據(jù)庫死鎖的相關(guān)問題
MYSQL數(shù)據(jù)庫1、死鎖發(fā)生的條件
MYSQL數(shù)據(jù)庫互斥條件:一個資源每次只能被一個進程使用;哀求與保持條件:一個進程因哀求資源而阻塞時,對已獲得的資源保持不放;不剝奪條件:進程已獲得的資源,在末使用完之前,不能強行剝奪;循環(huán)等待條件:若干進程之間形成一種頭尾相接的循環(huán)等待資源關(guān)系.
MYSQL數(shù)據(jù)庫2、死鎖檢測(根據(jù)網(wǎng)上的經(jīng)驗)
MYSQL數(shù)據(jù)庫Innodb檢測死鎖有兩種情況,一種是滿足循環(huán)等待條件,還有另一種策略:鎖結(jié)構(gòu)超過mysql配置中設(shè)置的最大數(shù)量或鎖的遍歷深度超過設(shè)置的最大深度時,innodb也會判斷為死鎖(這是提高性能方面的考慮,避免事務(wù)一次占用太多的資源).
MYSQL數(shù)據(jù)庫因循環(huán)等待條件而產(chǎn)生的死鎖只有可能是四種形式:兩張表兩行記錄交叉申請互斥鎖、同一張表則存在主鍵索引鎖沖突、主鍵索引鎖與非聚簇索引鎖沖突、鎖升級導(dǎo)致的鎖等待隊列阻塞.
MYSQL數(shù)據(jù)庫3、死鎖避免(根據(jù)網(wǎng)上的經(jīng)驗)
MYSQL數(shù)據(jù)庫1.如果使用insert…select語句備份表格且數(shù)據(jù)量較大,在單獨的時間點操作,避免與其他sql語句爭奪資源,或使用select into outfile加上load data infile代替 insert…select,這樣不僅快,而且不會要求鎖定
2. 一個鎖定記錄集的事務(wù),其操作結(jié)果集應(yīng)盡量簡短,以免一次占用太多資源,與其他事務(wù)處理的記錄沖突.
3.更新或者刪除表格數(shù)據(jù),sql語句的where條件都是主鍵或都是索引,避免兩種情況交叉,造成死鎖.對于where子句較復(fù)雜的情況,將其單獨通過sql得到后,再在更新語句中使用.
4. sql語句的嵌套表格不要太多,能拆分就拆分,避免占有資源同時等待資源,導(dǎo)致與其他事務(wù)沖突.
5. 對定點運行腳本的情況,避免在同一時間點運行多個對同一表進行讀寫的腳本,特別注意加鎖且操作數(shù)據(jù)量比擬大的語句.
6.應(yīng)用程序中增加對死鎖的判斷,如果事務(wù)意外結(jié)束,重新運行該事務(wù),減少對功能的影響.
MYSQL數(shù)據(jù)庫4、死鎖辦理
MYSQL數(shù)據(jù)庫1)先執(zhí)行show processlist找到死鎖線程號.然后Kill pid
MYSQL數(shù)據(jù)庫2)Show innodb status檢查引擎狀態(tài) ,可以看到哪些語句產(chǎn)生死鎖
MYSQL數(shù)據(jù)庫3)查看information_schema架構(gòu)下的innodb_locks、innodb_trx、innodb_lock_waits等表
MYSQL數(shù)據(jù)庫?
PS:Mysql死鎖
MYSQL數(shù)據(jù)庫既然談到死鎖,那附帶地就專門說一下.
何為死鎖?
?
死鎖是對資源的分配和使用不當而造成的.是兩個進程爭奪某一資源而出現(xiàn)相互等待的現(xiàn)象.具體的來講,出現(xiàn)死鎖需要滿足四個必要條件:
(1)互斥條件:每一個資源都只能被一個進程使用
(2)哀求與保持條件:一個進程因哀求資源而阻塞時,對已獲得的資源保持不放
(3)不剝奪條件:進程已獲得的資源,在末使用完之前,不能強行剝奪.
(4)循環(huán)等待條件:若干進程之間形成一種頭尾相接的循環(huán)等待資源關(guān)系.
很顯然,出現(xiàn)死鎖需要兩個或者兩個以上的進程,換句話說,死鎖發(fā)生在并發(fā)的程序中.在Mysql中,由于目前只有InnoDB引擎使用事務(wù)(InnoDB支持鎖),便有了InnoDB和死鎖的曠世基情.
死鎖的檢測
?
1、通過使用Show innodb status檢查引擎狀態(tài) ,可以看到哪些語句產(chǎn)生deadlock
2、MySQL提供了一個information_schema,通過查看innodb_locks、innodb_trx、innodb_lock_waits這幾個表檢測死鎖.
因循環(huán)等待條件而產(chǎn)生的死鎖只有可能是四種形式:兩張表兩行記錄交叉申請互斥鎖、同一張表則存在主鍵索引鎖沖突、主鍵索引鎖與非聚簇索引鎖沖突、鎖升級導(dǎo)致的鎖等待隊列阻塞.
MYSQL數(shù)據(jù)庫死鎖避免
MYSQL數(shù)據(jù)庫1.如果使用insert…select語句備份表格且數(shù)據(jù)量較大,在單獨的時間點操作,避免與其他sql語句爭奪資源,或使用select into outfile加上load data infile代替 insert…select,這樣不僅快,而且不會要求鎖定
2. 一個鎖定記錄集的事務(wù),其操作結(jié)果集應(yīng)盡量簡短,以免一次占用太多資源,與其他事務(wù)處理的記錄沖突.
3.更新或者刪除表格數(shù)據(jù),sql語句的where條件都是主鍵或都是索引,避免兩種情況交叉,造成死鎖.對于where子句較復(fù)雜的情況,將其單獨通過sql得到后,再在更新語句中使用.
4. sql語句的嵌套表格不要太多,能拆分就拆分,避免占有資源同時等待資源,導(dǎo)致與其他事務(wù)沖突.
5. 對定點運行腳本的情況,避免在同一時間點運行多個對同一表進行讀寫的腳本,特別注意加鎖且操作數(shù)據(jù)量比擬大的語句.
6.應(yīng)用程序中增加對死鎖的判斷,如果事務(wù)意外結(jié)束,重新運行該事務(wù),減少對功能的影響.
《Mysql學(xué)習MySQL中InnoDB存儲引擎的鎖的基本使用教程》是否對您有啟發(fā),歡迎查看更多與《Mysql學(xué)習MySQL中InnoDB存儲引擎的鎖的基本使用教程》相關(guān)教程,學(xué)精學(xué)透。維易PHP學(xué)院為您提供精彩教程。
轉(zhuǎn)載請注明本頁網(wǎng)址:
http://www.fzlkiss.com/jiaocheng/12363.html