前言應用場景java
事務必須知足傳統事務的特性,即原子性,一致性,分離性和持久性。可是分佈式事務處理過程當中,spring
某些場地好比在電商系統中,當有用戶下單後,除了在訂單表插入一條記錄外,對應商品表的這個商品數量必須減1吧,怎麼保證?數據庫
在搜索廣告系統中,當用戶點擊某廣告後,除了在點擊事件表中增長一條記錄外,
還得去商家帳戶表中找到這個商家並扣除廣告費吧,怎麼保證?服務器
一 本地事務
以用戶A轉帳用戶B爲例,假設有網絡
用戶A帳戶表:A(id,userId,amount) 架構
用戶B帳戶表:B(id,userId,amount)併發
用戶的userId=1;app
從用戶A轉帳1萬塊錢到用戶B的動做分爲兩步:分佈式
1)用戶A表扣除1萬:update A set amount=amount-10000 where userId=1;高併發
2)用戶B表增長1萬:update B set amount=amount+10000 where userId=1;
如何確保用戶A用戶B收支平衡呢?有人說這個很簡單嘛,能夠用事務解決。
Begin transaction update A set amount=amount-10000 where userId=1; update B set amount=amount+10000 where userId=1; End transaction commit;
很是正確!若是你使用spring的話一個註解就能搞定上述事務功能。
@Transactional(rollbackFor=Exception.class) public void update() { updateATable(); //更新A表 updateBTable(); //更新B表 }
若是系統規模較小,數據表都在一個數據庫實例上,上述本地事務方式能夠很好地運行,可是若是系統規模較大,
好比用戶A帳戶表和用戶B帳戶表顯然不會在同一個數據庫實例上,他們每每分佈在不一樣的物理節點上,這時本地事務已經失去用武之地。
既然本地事務失效,分佈式事務天然就登上舞臺。
二 XA
XA是由X/Open組織提出的分佈式事務的規範。XA規範主要 定義了(全局)事務管理器(Transaction Manager)和(局部)資源管理器(Resource Manager)之間的接口。
XA接口是雙向的系統接口,在事務管理器(Transaction Manager)以及一個或多個資源管理器(Resource Manager)之間造成通訊橋樑。
XA之因此須要引入事務管理器是由於,在分佈式系統中,從理論上講(參考Fischer等的論文),兩臺機器理論上無 法達到一致的狀態,須要引入一個單點進行協調。
事務管理器控制着全局事務,管理事務生命週期,並協調資源。資源管理器負責控制和管理實際資源(如數據庫或 JMS隊列)。
下圖說明了事務管理器、資源管理器,與應用程序之間的關係:
三 兩階段提交協議
分佈式事務必須知足傳統事務的特性,即原子性,一致性,分離性和持久性。可是分佈式事務處理過程當中,某些場地(Server)可能發生故障,
或 者因爲網絡發生故障而沒法訪問到某些場地。爲了防止分佈式系統部分失敗時產生數據的不一致性。
在分佈式事務的控制中採用了兩階段提交協議(Two- Phase Commit Protocol)。即事務的提交分爲兩個階段:
預提交階段(Pre-Commit Phase)
決策後階段(Post-Decision Phase)
兩階段提交用來協調參與一個更新中的多個服務器的活動,以防止分佈式系統部分失敗時產生數據的不一致性。例如,若是一個更新操做要求位於三個不一樣結點上的記錄被改變,且其中只要有一個結點失敗,另外兩個結點必須檢測到這個失敗並取消它們所作的改變。
爲了支持兩階段提交,一個分佈式更新事務中涉及到的服務器必須可以相互通訊。通常來講一個服務器會被指定爲"控制"或"提交"服務器並監控來自其它服務器的信息。
在分佈式更新期間,各服務器首先標誌它們已經完成(但未提交)指定給它們的分佈式事務的那一部分,並準備提交(以使它們的更新部分紅爲永久性的)。這是 兩階段提交的第一階段。若是有一結點不能響應,那麼控制服務器要指示其它結點撤消分佈式事務的各個部分的影響。若是全部結點都回答準備好提交,控制服務器 則指示它們提交併等待它們的響應。等待確認信息階段是第二階段。
在接收到能夠提交指示後,每一個服務器提交分佈式事務中屬於本身的那一部分,並給控制服務器 發回提交完成信息。
在一個分佈式事務中,必須有一個場地的Server做爲協調者(coordinator),它能向 其它場地的Server發出請求,並對它們的回答做出響應,由它來控制一個分佈式事務的提交或撤消。該分佈式事務中涉及到的其它場地的Server稱爲參 與者(Participant)。
事務兩階段提交的過程以下:
● 兩階段提交在應用程序向協調者發出一個提交命令時被啓動。這時提交進入第一階段,即預提交階段。在這一階段中:
(1) 協調者準備局部(即在本地)提交併在日誌中寫入"預提交"日誌項,幷包含有該事務的全部參與者的名字。
(2) 協調者詢問參與者可否提交該事務。一個參與者可能因爲多種緣由不能提交。例如,該Server提供的約束條件(Constraints)的延遲檢查不符合 限制條件時,不能提交;參與者自己的Server進程或硬件發生故障,不能提交;或者協調者訪問不到某參與者(網絡故障),這時協調者都認爲是收到了一個 否認的回答。
(3) 若是參與者可以提交,則在其自己的日誌中寫入"準備提交"日誌項,該日誌項當即寫入硬盤,而後給協調者發回,已準備好提交"的回答。
(4) 協調者等待全部參與者的回答,若是有參與者發回否認的回答,則協調者撤消該事務並給全部參與者發出一個"撤消該事務"的消息,結束該分佈式事務,撤消該事務的全部影響。
● 若是全部的參與者都送回"已準備好提交"的消息,則該事務的提交進入第二階段,即決策後提交階段。在這一階段中:
(1) 協調者在日誌中寫入"提交"日誌項,並當即寫入硬盤。
(2) 協調者向參與者發出"提交該事務"的命令。各參與者接到該命令後,在各自的日誌中寫入"提交"日誌項,並當即寫入硬盤。而後送回"已提交"的消息,釋放該事務佔用的資源。
(3) 當全部的參與者都送回"已提交"的消息後,協調者在日誌中寫入"事務提交完成"日誌項,釋放協調者佔用的資源 。這樣,完成了該分佈式事務的提交。
現現在實現基於兩階段提交的分佈式事務也沒那麼困難了,若是使用java,那麼可使用開源軟件atomikos來快速實現。
缺點
不過但凡使用過的上述兩階段提交的同窗均可以發現性能實在是太差,根本不適合高併發的系統。爲何?
1)兩階段提交涉及屢次節點間的網絡通訊,通訊時間太長!
2)事務時間相對於變長了,鎖定的資源的時間也變長了,形成資源等待時間也增長好多。
四 使用消息隊列來避免分佈式事務
若是仔細觀察生活的話,生活的不少場景已經給了咱們提示。
好比在北京頗有名的姚記炒肝點了炒肝並付了錢後,他們並不會直接把你點的炒肝給你,每每是給你一張小票,而後讓你拿着小票到出貨區排隊去取。
爲何他們要將付錢和取貨兩個動做分開呢?緣由不少,其中一個很重要的緣由是爲了使他們接待能力加強(併發量更高)。
仍是回到咱們的問題,只要這張小票在,你最終是能拿到炒肝的。同理轉帳服務也是如此,當用戶A帳戶扣除1萬後,
咱們只要生成一個憑證(消息)便可,這個憑證(消息)上寫着「讓用戶B帳戶增長 1萬」,只要這個憑證(消息)能可靠保存,
咱們最終是能夠拿着這個憑證(消息)讓用戶B帳戶增長1萬的,即咱們能依靠這個憑證(消息)完成最終一致性。
4.1 如何可靠保存憑證(消息)
有兩種方法:
4.1.1 業務與消息耦合的方式
用戶A在完成扣款的同時,同時記錄消息數據,這個消息數據與業務數據保存在同一數據庫實例裏(消息記錄表表名爲message);
Begin transaction update A set amount=amount-10000 where userId=1; insert into message(userId, amount,status) values(1, 10000, 1); End transaction commit;
上述事務能保證只要用戶A帳戶裏被扣了錢,消息必定能保存下來。
當上述事務提交成功後,咱們經過實時消息服務將此消息通知用戶B,用戶B處理成功後發送回覆成功消息,用戶A收到回覆後刪除該條消息數據。
4.1.2 業務與消息解耦方式
上述保存消息的方式使得消息數據和業務數據緊耦合在一塊兒,從架構上看不夠優雅,並且容易誘發其餘問題。爲了解耦,能夠採用如下方式。
1)用戶A在扣款事務提交以前,向實時消息服務請求發送消息,實時消息服務只記錄消息數據,而不真正發送,只有消息發送成功後纔會提交事務;
2)當用戶A扣款事務被提交成功後,向實時消息服務確認發送。只有在獲得確認發送指令後,實時消息服務才真正發送該消息;
3)當用戶A扣款事務提交失敗回滾後,向實時消息服務取消發送。在獲得取消發送指令後,該消息將不會被髮送;
4)對於那些未確認的消息或者取消的消息,須要有一個消息狀態確認系統定時去用戶A系統查詢這個消息的狀態並進行更新。爲何須要這一步驟,
舉個例子:假設在第2步用戶A扣款事務被成功提交後,系統掛了,此時消息狀態並未被更新爲「確認發送」,從而致使消息不能被髮送。
優勢:消息數據獨立存儲,下降業務系統與消息系統間的耦合;
缺點:一次消息發送須要兩次請求;業務處理服務須要實現消息狀態回查接口。
4.2 如何解決消息重複投遞的問題
還有一個很嚴重的問題就是消息重複投遞,以咱們用戶A轉帳到用戶B爲例,若是相同的消息被重複投遞兩次,那麼咱們用戶B帳戶將會增長2萬而不是1萬了。
爲何相同的消息會被重複投遞?好比用戶B處理完消息msg後,發送了處理成功的消息給用戶A,正常狀況下用戶A應該要刪除消息msg,但若是用戶A這時候悲劇的掛了,
重啓後一看消息msg還在,就會繼續發送消息msg。
解決方法很簡單,在用戶B這邊增長消息應用狀態表(message_apply),通俗來講就是個帳本,用於記錄消息的消費狀況,每次來一個消息,
在真正執行以前,先去消息應用狀態表中查詢一遍,若是找到說明是重複消息,丟棄便可,若是沒找到才執行,同時插入到消息應用狀態表(同一事務)。
for each msg in queue Begin transaction select count(*) as cnt from message_apply where msg_id=msg.msg_id; if cnt==0 then update B set amount=amount+10000 where userId=1; insert into message_apply(msg_id) values(msg.msg_id); End transaction commit;