并發(fā)編程的原子性 != 事務(wù)ACID的原子性
關(guān)于原子性,很多人在多個(gè)地方都聽說過,大家也都背的很熟悉。
在事務(wù)的ACID中,有原子性的概念,在并發(fā)編程的原子性、可見性、有序性中也有原子性的概念。?
有人認(rèn)為他們是一樣的,甚至很多人在說原子性的時(shí)候也都是這么說的:?
?原子性是指事務(wù)是一個(gè)不可再分割的工作單元,事務(wù)中的操作要么都發(fā)生,要么都不發(fā)生。
但是,其實(shí),雖然都叫原子性,但是此原子性非彼原子性。
在數(shù)據(jù)庫(kù)中的原子性,確實(shí)是一個(gè)不可拆分的工作單元,要么都執(zhí)行,要么都不執(zhí)行。因?yàn)槭聞?wù)可以Commit、也可以Rollback。?
但是在并發(fā)編程中,一個(gè)操作是沒辦法rollback的,并且線程在執(zhí)行過程中也是有可能失敗的,失敗了是沒辦法回滾的,難道就說一個(gè)操作沒辦法保證原子性了么?
所以,在并發(fā)編程中,我們把一個(gè)或者多個(gè)操作在 CPU 執(zhí)行的過程中不被中斷的特性稱為原子性。這里的原子性是通過加鎖的方式來保證的,其實(shí)保證的就是一系列操作,不可以被拆分執(zhí)行,即執(zhí)行過程中,需要互斥排他,不能有其他線程進(jìn)行執(zhí)行。
舉一個(gè)例子來說明一下這兩個(gè)原子性的區(qū)別。
Redis中的Lua腳本到底能不能保證原子性??
網(wǎng)上很多文章,有人說能,有人說不能。要我說,都對(duì),也都不對(duì)。就是因?yàn)橐驗(yàn)榇蠹腋慊炝诉@兩個(gè)原子性的區(qū)別。
我們都知道,當(dāng)我們想要在一個(gè)事務(wù)中執(zhí)行多個(gè)命令的時(shí)候,會(huì)選擇使用Lua腳本。
Redis會(huì)將一個(gè)要執(zhí)行的Lua腳本封裝成一個(gè)單獨(dú)的事務(wù),而腳本執(zhí)行器在執(zhí)行這個(gè)事務(wù)的過程中,
如果有其他客戶端請(qǐng)求的時(shí)候,會(huì)把它暫存起來,等腳本處理完以后,才會(huì)再把被暫存的請(qǐng)求恢復(fù)執(zhí)行。
這樣就可以保證整個(gè)腳本是作為一個(gè)整體執(zhí)行的,中間不會(huì)被其他命令插入,這就是所謂的原子性中的”不可拆分”特性。
但是,如果事務(wù)執(zhí)行過程中命令產(chǎn)生錯(cuò)誤,事務(wù)是不會(huì)回滾的,也不會(huì)影響后續(xù)命令的執(zhí)行。
也就是說,Redis保證以原子方式執(zhí)行Lua腳本,但是不保證腳本中所有操作要么都執(zhí)行或者都會(huì)滾。
那就意味著,Redis中Lua腳本的執(zhí)行,可以保證并發(fā)編程中不可再拆分的這個(gè)原子性,但是沒有保證數(shù)據(jù)庫(kù)ACID中要么都執(zhí)行要么都會(huì)滾的這個(gè)原子性。