欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

mysql悲觀鎖怎么實現(xiàn) mysql如何實現(xiàn)悲觀鎖

mysql的事務四個特性以及事務的四個隔離級別

分別是原子性、一致性、隔離性、持久性。

創(chuàng)新互聯(lián)公司主營威信網(wǎng)站建設的網(wǎng)絡公司,主營網(wǎng)站建設方案,APP應用開發(fā),威信h5小程序開發(fā)搭建,威信網(wǎng)站營銷推廣歡迎威信等地區(qū)企業(yè)咨詢

原子性是指事務包含的所有操作要么全部成功,要么全部失敗回滾,因此事務的操作如果成功就必須要完全應用到數(shù)據(jù)庫,如果操作失敗則不能對數(shù)據(jù)庫有任何影響。

一致性是指事務必須使數(shù)據(jù)庫從一個一致性狀態(tài)變換到另一個一致性狀態(tài),也就是說一個事務執(zhí)行之前和執(zhí)行之后都必須處于一致性狀態(tài)。舉例來說,假設用戶A和用戶B兩者的錢加起來一共是1000,那么不管A和B之間如何轉(zhuǎn)賬、轉(zhuǎn)幾次賬,事務結(jié)束后兩個用戶的錢相加起來應該還得是1000,這就是事務的一致性。

隔離性是當多個用戶并發(fā)訪問數(shù)據(jù)庫時,比如同時操作同一張表時,數(shù)據(jù)庫為每一個用戶開啟的事務,不能被其他事務的操作所干擾,多個并發(fā)事務之間要相互隔離。關于事務的隔離性數(shù)據(jù)庫提供了多種隔離級別,稍后會介紹到。

持久性是指一個事務一旦被提交了,那么對數(shù)據(jù)庫中的數(shù)據(jù)的改變就是永久性的,即便是在數(shù)據(jù)庫系統(tǒng)遇到故障的情況下也不會丟失提交事務的操作。例如我們在使用JDBC操作數(shù)據(jù)庫時,在提交事務方法后,提示用戶事務操作完成,當我們程序執(zhí)行完成直到看到提示后,就可以認定事務已經(jīng)正確提交,即使這時候數(shù)據(jù)庫出現(xiàn)了問題,也必須要將我們的事務完全執(zhí)行完成。否則的話就會造成我們雖然看到提示事務處理完畢,但是數(shù)據(jù)庫因為故障而沒有執(zhí)行事務的重大錯誤。這是不允許的。

在數(shù)據(jù)庫操作中,在并發(fā)的情況下可能出現(xiàn)如下問題:

正是為了解決以上情況,數(shù)據(jù)庫提供了幾種隔離級別。

數(shù)據(jù)庫事務的隔離級別有4個,由低到高依次為Read uncommitted(未授權(quán)讀取、讀未提交)、Read committed(授權(quán)讀取、讀提交)、Repeatable read(可重復讀取)、Serializable(序列化),這四個級別可以逐個解決臟讀、不可重復讀、幻象讀這幾類問題。

雖然數(shù)據(jù)庫的隔離級別可以解決大多數(shù)問題,但是靈活度較差,為此又提出了悲觀鎖和樂觀鎖的概念。

悲觀鎖,它指的是對數(shù)據(jù)被外界(包括本系統(tǒng)當前的其他事務,以及來自外部系統(tǒng)的事務處理)修改持保守態(tài)度。因此,在整個數(shù)據(jù)處理過程中,將數(shù)據(jù)處于鎖定狀態(tài)。悲觀鎖的實現(xiàn),往往依靠數(shù)據(jù)庫提供的鎖機制。也只有數(shù)據(jù)庫層提供的鎖機制才能真正保證數(shù)據(jù)訪問的排他性,否則,即使在本系統(tǒng)的數(shù)據(jù)訪問層中實現(xiàn)了加鎖機制,也無法保證外部系統(tǒng)不會修改數(shù)據(jù)。

商品t_items表中有一個字段status,status為1代表商品未被下單,status為2代表商品已經(jīng)被下單(此時該商品無法再次下單),那么我們對某個商品下單時必須確保該商品status為1。假設商品的id為1。

如果不采用鎖,那么操作方法如下:

但是上面這種場景在高并發(fā)訪問的情況下很可能會出現(xiàn)問題。例如當?shù)谝徊讲僮髦?,查詢出來的商品status為1。但是當我們執(zhí)行第三步Update操作的時候,有可能出現(xiàn)其他人先一步對商品下單把t_items中的status修改為2了,但是我們并不知道數(shù)據(jù)已經(jīng)被修改了,這樣就可能造成同一個商品被下單2次,使得數(shù)據(jù)不一致。所以說這種方式是不安全的。

在上面的場景中,商品信息從查詢出來到修改,中間有一個處理訂單的過程,使用悲觀鎖的原理就是,當我們在查詢出t_items信息后就把當前的數(shù)據(jù)鎖定,直到我們修改完畢后再解鎖。那么在這個過程中,因為t_items被鎖定了,就不會出現(xiàn)有第三者來對其進行修改了。需要注意的是,要使用悲觀鎖,我們必須關閉mysql數(shù)據(jù)庫的自動提交屬性,因為MySQL默認使用autocommit模式,也就是說,當你執(zhí)行一個更新操作后,MySQL會立刻將結(jié)果進行提交。我們可以使用命令設置MySQL為非autocommit模式: set autocommit=0;

設置完autocommit后,我們就可以執(zhí)行我們的正常業(yè)務了。具體如下:

上面的begin/commit為事務的開始和結(jié)束,因為在前一步我們關閉了mysql的autocommit,所以需要手動控制事務的提交。

上面的第一步我們執(zhí)行了一次查詢操作: select status from t_items where id=1 for update; 與普通查詢不一樣的是,我們使用了 select…for update 的方式,這樣就通過數(shù)據(jù)庫實現(xiàn)了悲觀鎖。此時在t_items表中,id為1的那條數(shù)據(jù)就被我們鎖定了,其它的事務必須等本次事務提交之后才能執(zhí)行。這樣我們可以保證當前的數(shù)據(jù)不會被其它事務修改。需要注意的是,在事務中,只有 SELECT ... FOR UPDATE 或 LOCK IN SHARE MODE 操作同一個數(shù)據(jù)時才會等待其它事務結(jié)束后才執(zhí)行,一般 SELECT ... 則不受此影響。拿上面的實例來說,當我執(zhí)行 select status from t_items where id=1 for update; 后。我在另外的事務中如果再次執(zhí)行 select status from t_items where id=1 for update; 則第二個事務會一直等待第一個事務的提交,此時第二個查詢處于阻塞的狀態(tài),但是如果我是在第二個事務中執(zhí)行 select status from t_items where id=1; 則能正常查詢出數(shù)據(jù),不會受第一個事務的影響。

使用 select…for update 會把數(shù)據(jù)給鎖住,不過我們需要注意一些鎖的級別,MySQL InnoDB默認Row-Level Lock,所以只有「明確」地指定主鍵或者索引,MySQL 才會執(zhí)行Row lock (只鎖住被選取的數(shù)據(jù)) ,否則MySQL 將會執(zhí)行Table Lock (將整個數(shù)據(jù)表單給鎖住)。舉例如下:

1、 select * from t_items where id=1 for update;

這條語句明確指定主鍵(id=1),并且有此數(shù)據(jù)(id=1的數(shù)據(jù)存在),則采用row lock。只鎖定當前這條數(shù)據(jù)。

2、 select * from t_items where id=3 for update;

這條語句明確指定主鍵,但是卻查無此數(shù)據(jù),此時不會產(chǎn)生lock(沒有元數(shù)據(jù),又去lock誰呢?)。

3、 select * from t_items where name='手機' for update;

這條語句沒有指定數(shù)據(jù)的主鍵,那么此時產(chǎn)生table lock,即在當前事務提交前整張數(shù)據(jù)表的所有字段將無法被查詢。

4、 select * from t_items where id0 for update; 或者 select * from t_items where id1 for update; (注:在SQL中表示不等于)

上述兩條語句的主鍵都不明確,也會產(chǎn)生table lock。

5、 select * from t_items where status=1 for update; (假設為status字段添加了索引)

這條語句明確指定了索引,并且有此數(shù)據(jù),則產(chǎn)生row lock。

6、 select * from t_items where status=3 for update; (假設為status字段添加了索引)

這條語句明確指定索引,但是根據(jù)索引查無此數(shù)據(jù),也就不會產(chǎn)生lock。

樂觀鎖( Optimistic Locking ) 相對悲觀鎖而言,樂觀鎖假設認為數(shù)據(jù)一般情況下不會造成沖突,所以只會在數(shù)據(jù)進行提交更新的時候,才會正式對數(shù)據(jù)的沖突與否進行檢測,如果發(fā)現(xiàn)沖突了,則返回用戶錯誤的信息,讓用戶決定如何去做。實現(xiàn)樂觀鎖一般來說有以下2種方式:

Mysql中鎖的類型有哪些呢?

mysql鎖分為共享鎖和排他鎖,也叫做讀鎖和寫鎖。

讀鎖是共享的,可以通過lock in share mode實現(xiàn),這時候只能讀不能寫。

寫鎖是排他的,它會阻塞其他的寫鎖和讀鎖。從顆粒度來區(qū)分,可以分為表鎖和?鎖兩種。

表鎖會鎖定整張表并且阻塞其他?戶對該表的所有讀寫操作,?如alter修改表結(jié)構(gòu)的時候會鎖表。

?鎖?可以分為樂觀鎖和悲觀鎖,悲觀鎖可以通過for update實現(xiàn),樂觀鎖則通過版本號實現(xiàn)。

mysql中的樂觀鎖和悲觀鎖怎么用

關于mysql中的樂觀鎖和悲觀鎖面試的時候被問到的概率還是比較大的。

mysql的悲觀鎖:

其實理解起來非常簡單,當數(shù)據(jù)被外界修改持保守態(tài)度,包括自身系統(tǒng)當前的其他事務,以及來自外部系統(tǒng)的事務處理,因此,在整個數(shù)據(jù)處理過程中,將數(shù)據(jù)處于鎖定狀態(tài)。悲觀鎖的實現(xiàn),往往依靠數(shù)據(jù)庫提供的鎖機制,但是也只有數(shù)據(jù)庫層提供的鎖機制才能真正保證數(shù)據(jù)訪問的排他性,否則,即使在自身系統(tǒng)中實現(xiàn)了加鎖機制,也無法保證外部系統(tǒng)不會修改數(shù)據(jù)。

來點實際的,當我們使用悲觀鎖的時候我們首先必須關閉mysql數(shù)據(jù)庫的自動提交屬性,因為MySQL默認使用autocommit模式,也就是說,當你執(zhí)行一個更新操作后,MySQL會立刻將結(jié)果進行提交。

關閉命令為:set autocommit=0;

悲觀鎖可以使用select…for update實現(xiàn),在執(zhí)行的時候會鎖定數(shù)據(jù),雖然會鎖定數(shù)據(jù),但是不影響其他事務的普通查詢使用。此處說普通查詢就是平時我們用的:select * from table 語句。在我們使用悲觀鎖的時候事務中的語句例如:

//開始事務

begin;/begin work;/start transaction; (三選一)

//查詢信息

select * from order where id=1 for update;

//修改信息

update order set name='names';

//提交事務

commit;/commit work;(二選一)

此處的查詢語句for update關鍵字,在事務中只有SELECT ... FOR UPDATE 或LOCK IN SHARE MODE 同一條數(shù)據(jù)時會等待其它事務結(jié)束后才執(zhí)行,一般的SELECT查詢則不受影響。

執(zhí)行事務時關鍵字select…for update會鎖定數(shù)據(jù),防止其他事務更改數(shù)據(jù)。但是鎖定數(shù)據(jù)也是有規(guī)則的。

查詢條件與鎖定范圍:

1、具體的主鍵值為查詢條件

比如查詢條件為主鍵ID=1等等,如果此條數(shù)據(jù)存在,則鎖定當前行數(shù)據(jù),如果不存在,則不鎖定。

2、不具體的主鍵值為查詢條件

比如查詢條件為主鍵ID1等等,此時會鎖定整張數(shù)據(jù)表。

3、查詢條件中無主鍵

會鎖定整張數(shù)據(jù)表。

4、如果查詢條件中使用了索引為查詢條件

明確指定索引并且查到,則鎖定整條數(shù)據(jù)。如果找不到指定索引數(shù)據(jù),則不加鎖。

悲觀鎖的確保了數(shù)據(jù)的安全性,在數(shù)據(jù)被操作的時候鎖定數(shù)據(jù)不被訪問,但是這樣會帶來很大的性能問題。因此悲觀鎖在實際開發(fā)中使用是相對比較少的。

mysql的樂觀鎖:

相對悲觀鎖而言,樂觀鎖假設數(shù)據(jù)一般情況下不會造成沖突,所以在數(shù)據(jù)進行提交更新的時候,才會對數(shù)據(jù)的沖突與否進行檢測,如果發(fā)現(xiàn)沖突,則讓返回用戶錯誤的信息,讓用戶決定如何去做。

一般來說,實現(xiàn)樂觀鎖的方法是在數(shù)據(jù)表中增加一個version字段,每當數(shù)據(jù)更新的時候這個字段執(zhí)行加1操作。這樣當數(shù)據(jù)更改的時候,另外一個事務訪問此條數(shù)據(jù)進行更改的話就會操作失敗,從而避免了并發(fā)操作錯誤。當然,還可以將version字段改為時間戳,不過原理都是一樣的。

例如有表student,字段:

id,name,version

1 a 1

當事務一進行更新操作:update student set name='ygz' where id = #{id} and version = #{version};

此時操作完后數(shù)據(jù)會變?yōu)閕d = 1,name = ygz,version = 2,當另外一個事務二同樣執(zhí)行更新操作的時候,卻發(fā)現(xiàn)version != 1,此時事務二就會操作失敗,從而保證了數(shù)據(jù)的正確性。

悲觀鎖和樂觀鎖都是要根據(jù)具體業(yè)務來選擇使用,本文僅作簡單介紹。

悲觀鎖和樂觀鎖定義是什么?

悲觀鎖和樂觀鎖定義:

樂觀鎖:樂觀鎖在操作數(shù)據(jù)時非常樂觀,認為別人不會同時修改數(shù)據(jù)。因此樂觀鎖不會上鎖,只是在執(zhí)行更新的時候判斷一下在此期間別人是否修改了數(shù)據(jù):如果別人修改了數(shù)據(jù)則放棄操作,否則執(zhí)行操作。

悲觀鎖:悲觀鎖在操作數(shù)據(jù)時比較悲觀,認為別人會同時修改數(shù)據(jù)。因此操作數(shù)據(jù)時直接把數(shù)據(jù)鎖住,直到操作完成后才會釋放鎖;上鎖期間其他人不能修改數(shù)據(jù)。

悲觀鎖實現(xiàn)方式

悲觀鎖的實現(xiàn),往往依靠數(shù)據(jù)庫提供的鎖機制。在數(shù)據(jù)庫中,悲觀鎖的流程如下:

1.在對記錄進行修改之前,先嘗試為該記錄加上排它鎖(exclusive locking)。

2.如果加鎖失敗,說明該記錄正在被修改,那么當前查詢可能要等待或者拋出異常。具體響應方式由開發(fā)者根據(jù)實際需要決定。

3.如果成功加鎖,那么就可以對記錄做修改,事務完成后就會解鎖了。

4.期間如果有其他對該記錄做修改或加排它鎖的操作,都會等待解鎖或直接拋出異常。

網(wǎng)站欄目:mysql悲觀鎖怎么實現(xiàn) mysql如何實現(xiàn)悲觀鎖
文章轉(zhuǎn)載:http://chinadenli.net/article40/doddpeo.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供ChatGPT、用戶體驗、搜索引擎優(yōu)化外貿(mào)網(wǎng)站建設、品牌網(wǎng)站設計定制開發(fā)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

外貿(mào)網(wǎng)站制作