原文鏈接:https://blog.ouyangsihai.cn/ >> MySQL事務(wù),這篇文章就夠了
創(chuàng)新互聯(lián)公司專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于網(wǎng)站設(shè)計(jì)、網(wǎng)站制作、武威網(wǎng)絡(luò)推廣、微信小程序定制開發(fā)、武威網(wǎng)絡(luò)營銷、武威企業(yè)策劃、武威品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運(yùn)營等,從售前售中售后,我們都將竭誠為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);創(chuàng)新互聯(lián)公司為所有大學(xué)生創(chuàng)業(yè)者提供武威建站搭建服務(wù),24小時(shí)服務(wù)熱線:13518219792,官方網(wǎng)址:www.cdcxhl.com
在看這篇文章之前,我們回顧一下前面的幾篇關(guān)于MySQL的系列文章,應(yīng)該對(duì)你讀下面的文章有所幫助。
事務(wù)(Transaction)是并發(fā)控制的基本單位。所謂的事務(wù),它是一個(gè)操作序列,這些操作要么都 執(zhí)行,要么都不執(zhí)行,它是一個(gè)不可分割的工作單位。事務(wù)是數(shù)據(jù)庫維護(hù)數(shù)據(jù)一致性的單位,在每 個(gè)事務(wù)結(jié)束時(shí),都能保持?jǐn)?shù)據(jù)一致性。
同時(shí),事務(wù)有著嚴(yán)格的地定義,必須滿足四個(gè)特性,也就是我們一直說的ACID,但是,并不是說各種數(shù)據(jù)庫就一定會(huì)滿足四個(gè)特性,對(duì)于不同的數(shù)據(jù)庫的實(shí)現(xiàn)來說,在不同程度上是不一定完全滿足要求的,比如,Oracle數(shù)據(jù)庫來說,默認(rèn)的事務(wù)隔離級(jí)別是READ COMMITTED
,是不滿足隔離性的要求的。
下面我們趁熱打鐵,介紹一下事務(wù)的必知必會(huì)的四大特性,這幾個(gè)特性也是在面試中,面試官面試MySQL的相關(guān)知識(shí)的時(shí)候,問的比較多的問題,所以,這幾個(gè)特性務(wù)必需要理解并且透徹的記在心里,開個(gè)玩笑,被火車撞了,也不應(yīng)該忘記這四個(gè)特性!
事務(wù)的四大特性簡稱為:ACID
,分別是原子性、一致性、隔離性和持久性。
下面我們一一來介紹一下。
原子性指的是整個(gè)數(shù)據(jù)庫的事務(wù)是一個(gè)不可分割的工作單位,每一個(gè)都應(yīng)該是一個(gè)原子操作。
當(dāng)我們執(zhí)行一個(gè)事務(wù)的時(shí)候,如果一系列的操作中,有一個(gè)操作失敗了,那么,需要將這一個(gè)事務(wù)中的所有操作恢復(fù)到執(zhí)行事務(wù)之前的狀態(tài),這就是事務(wù)的原子性。
下面舉個(gè)簡單的例子。
i++;
上面這個(gè)最簡單不過的代碼經(jīng)常也會(huì)被問到,這是一個(gè)原子操作嗎?那肯定不是,如果我們把這個(gè)代碼放到一個(gè)事務(wù)中來說,當(dāng)i+1
出現(xiàn)問題的時(shí)候,回滾的就是整個(gè)代碼i++(i = i + 1)了,所以回滾之后,i的值也是不會(huì)改變的。
以上就是原子性的概念。
一致性是指事務(wù)將數(shù)據(jù)庫從一種狀態(tài)轉(zhuǎn)變?yōu)橄乱环N一致性的狀態(tài),也就是說在事務(wù)執(zhí)行前后,這兩種狀態(tài)應(yīng)該是一樣的,也就是數(shù)據(jù)庫的完整性約束不會(huì)被破壞。
另外,需要注意的是一致性是×××,比如銀行轉(zhuǎn)賬的過程,你轉(zhuǎn)賬給別人,至于中間的狀態(tài),你少了500 ,他多了500,這些中間狀態(tài)×××次轉(zhuǎn)賬中間狀態(tài)也是不可見的,只有最后的成功或者失敗的狀態(tài)是可見的。
如果到分布式的一致性問題,又可以分為強(qiáng)一致性、弱一致性和最終一致性,關(guān)于這些概念,可以自己查查,還是很有意思的。
事務(wù)我們是可以開啟很多的,MySQL數(shù)據(jù)庫中可以同時(shí)啟動(dòng)很多的事務(wù),但是,事務(wù)和事務(wù)之間他們是相互分離的,也就是互不影響的,這就是事務(wù)的隔離性。
事務(wù)的持久性是指事務(wù)一旦提交,就是永久的了,就是發(fā)生問題,數(shù)據(jù)庫也是可以恢復(fù)的。因此,持久性保證事務(wù)的高可靠性。
事務(wù)可以分為很多中類型,一般分為:扁平事務(wù)、帶有保存點(diǎn)的扁平事務(wù)、鏈?zhǔn)聞?wù)、嵌套事務(wù)、分布式事務(wù)。
扁平事務(wù)是最簡單的一種,在實(shí)際開發(fā)中也是使用的最多的一種事務(wù)。在這種事務(wù)中,所有操作都處于同一層次,最常見的方式如下:
BEGIN WORK
Operation 1
Operation 2
Operation 3
...
Operation N
COMMIT WORK
舉個(gè)例子
begin work;
select * from user;
update user set name = 'sihai' where id = 1;
commit work;
扁平事務(wù)的主要缺點(diǎn)是不能提交或回滾事務(wù)的某一部分,或者分幾個(gè)獨(dú)立的步驟去提交。
這種事務(wù)除了支持扁平事務(wù)支持的操作外,這種事務(wù)跟扁平事務(wù)最大的區(qū)別就是允許在事務(wù)執(zhí)行過程中回滾到同一事務(wù)中較早的一個(gè)狀態(tài),這是因?yàn)榭赡苣承┦聞?wù)在執(zhí)行過程中出現(xiàn)的錯(cuò)誤并不會(huì)對(duì)所有的操作都無效,放棄整個(gè)事務(wù)不合乎要求,開銷也太大。保存點(diǎn)用來通知系統(tǒng)應(yīng)該記住事務(wù)當(dāng)前的狀態(tài),以便以后發(fā)生錯(cuò)誤時(shí),事務(wù)能回到該狀態(tài)。
舉個(gè)例子
begin work;
select * from user;
savepoint t1;
update user set name = 'sihai' where id = 1;
savepoint t2;
commit work;
通過上面的方式我們就建立了兩個(gè)保存點(diǎn)t1、t2,通過ROLLBACK TO SAVEPOINT t1
,我們就可以返回到保存點(diǎn)t1
。
鏈?zhǔn)聞?wù):在提交一個(gè)事務(wù)時(shí),釋放不需要的數(shù)據(jù)對(duì)象,將必要的處理上下文隱式的傳給下一個(gè)要開始的事務(wù)。需要注意,提交事務(wù)操作和下一個(gè)事務(wù)操作將合并為一個(gè)原子操作,就是下一個(gè)事務(wù)可以看到上一個(gè)事務(wù)的結(jié)果。
鏈?zhǔn)聞?wù),就是指回滾時(shí),只能恢復(fù)到最近一個(gè)保存點(diǎn);而帶有保存點(diǎn)的扁平事務(wù)則可以回滾到任意正確的保存點(diǎn)。
舉個(gè)例子
begin work;
select * from user;
savepoint t1;
update user set name = 'sihai' where id = 1;
savepoint t2;
commit work;
還是這個(gè)例子,但是對(duì)于鏈?zhǔn)聞?wù)來說,是不能直接rollback到保存點(diǎn)t1的,最能恢復(fù)到最近的一個(gè)保存點(diǎn)t2;另外我們需要注意,鏈?zhǔn)聞?wù)在執(zhí)行commit后就會(huì)釋放當(dāng)前事務(wù)所持有的所有鎖,而帶有保存點(diǎn)的扁平事務(wù)不會(huì)影響所持有的鎖。
在事務(wù)中再嵌套事務(wù),這種結(jié)構(gòu)有點(diǎn)像一顆橫著的樹的結(jié)構(gòu),位于根節(jié)點(diǎn)的事務(wù)稱為頂層事務(wù)。事務(wù)的前驅(qū)稱為父事務(wù),其它事務(wù)稱為子事務(wù)。事務(wù)的前驅(qū)稱為父事務(wù),事務(wù)的下一層稱為子事務(wù)。
子事務(wù)既可以提交也可以回滾,但是它的提交操作并不馬上生效,除非由其父事務(wù)提交。因此就可以確定,任何子事務(wù)都在頂層事務(wù)提交后才真正的被提交了。同理,任意一個(gè)事務(wù)的回滾都會(huì)引起它的所有子事務(wù)一同回滾。
BEGIN WORK
SubTransaction1:
BEGIN WORK
SubOperationX
COMMIT WORK
SubTransaction2:
BEGIN WORK
SubOperationY
COMMIT WORK
...
SubTransactionN:
BEGIN WORK
SubOperationN
COMMIT WORK
COMMIT WORK
分布式事務(wù)通常是指在一個(gè)分布式環(huán)境下運(yùn)行的扁平事務(wù),因此需要根據(jù)數(shù)據(jù)所在位置訪問網(wǎng)絡(luò)中的不同節(jié)點(diǎn)。
在不同的物理地址,通過網(wǎng)絡(luò)訪問,執(zhí)行不同的事務(wù),這就是分布式事務(wù)。
首先這一部分我們還是先介紹一下這些事務(wù)的語句,也不是很多,使用也不復(fù)雜,下面用一個(gè)表格做一個(gè)整理。
注意:
COMMIT
和COMMIT WORK
語句不同之處在于COMMIT WORK用來控制事務(wù)結(jié)束后的行為是CHAIN
還是RELEASE
,如果是CHAIN,那么事務(wù)就是鏈?zhǔn)聞?wù)。
用戶可以通過參數(shù)completion_type
控制,如下:
執(zhí)行下面的操作;
SET @@completion_type = 1;
BEGIN WORK;
INSERT INTO lock_test SELECT 10;
COMMIT WORK;
接著我們再執(zhí)行下面的操作;
INSERT INTO lock_test SELECT 115;
ROLLBACK;
SELECT * FROM lock_test;
我們先插入一條數(shù)據(jù)115,然后再回滾,我們知道如果不是在一個(gè)事務(wù)的時(shí)候,115應(yīng)該是會(huì)插入成功的,就算我們回滾了,但是,這里我們回滾之后,查詢結(jié)果如下:
這個(gè)時(shí)候并沒有115這條記錄,也就是回滾生效了,說明在COMMIT WORK
之后,又是一個(gè)新的事務(wù),所以才會(huì)出現(xiàn)這樣的結(jié)果。
我們先進(jìn)行下面的操作;
SET @@completion_type = 2;
BEGIN WORK;
INSERT INTO lock_test SELECT 5;
COMMIT WORK;
上面我們已經(jīng)提交事務(wù)了,當(dāng)我們使用下面的語句查詢lock_test的數(shù)據(jù)的時(shí)候,就會(huì)出現(xiàn)斷開連接。
SELECT * FROM lock_test;
事務(wù)的隔離級(jí)別有四種分別是:
對(duì)于這幾種隔離級(jí)別會(huì)帶來的問題及總結(jié),可以查看這篇文章:MySQL的又一神器-鎖,MySQL面試必備
這篇文章從下面幾個(gè)內(nèi)容介紹了一下MySQL數(shù)據(jù)庫事務(wù)的內(nèi)容,更詳細(xì)的其他內(nèi)容在后面的文章中再講解。
文章有不當(dāng)之處,歡迎指正,如果喜歡微信閱讀,你也可×××學(xué)java`,獲取優(yōu)質(zhì)學(xué)習(xí)資源。