JPA和分佈式事務簡介

1. Transaction 分兩種,Local Transaction 和 Global Transaction。 
涉及到一個Connection的Commit,稱爲Local Transaction。 
涉及到多個Connection的Commit,稱爲Global Transaction。 
樓主提到的是,Global Transaction. 

2. Global Transaction 須要XA接口(包括在JTA裏面)的支持。 

import javax.sql.XAConnection; 
import javax.transaction.xa.Xid; 
import javax.transaction.xa.XAResource; 
import javax.transaction.xa.XAException; 
import javax.transaction.Transaction; 
import javax.transaction.TransactionManager; 

其中的 
javax.sql.XAConnection; 
javax.transaction.xa.Xid; 
javax.transaction.xa.XAResource; 

這些XA接口的實現,須要數據庫的JDBC提供。 
數據庫自己要支持XA。數據庫的JDBC也要提供XA的實現。 

Oracle, Sybase, DB2, SQL Server等大型數據庫才支持XA, 支持Global Transaction。 
My SQL 連Local Transaction都支持很差,更別說Global Transation了。 

3. XA須要兩階段提交 -- prepare 和 commit. 
假設有兩個Connection, con1, con2, 大致的過程以下, java

con1 = XAResouce1.getConnection...
con2 = XAResouce2.getConnection...

con1 do some thing.
con2 do some thing.
after they finish.

pre1 = XAResouce1.prepare();
pre2 = XAResouce2.prepare();

if( both pre1 and pre2 are OK){
XAResouce1 and 2 commit
}else {
XAResouce1 and 2 rollback
}

前面有人講了,在XAResouce1 and 2 commit的時候, 
可能XAResouce1 commit() 成功了,XAResouce2 commit()失敗了。 
這時候,會拋出一個 「啓發式異常」。程序能夠處理這個異常。好比,XAResouce.recover()之類。 
但通常狀況下,還真沒別的辦法,須要數據管理員根據數據操做日誌 undo全部的操做,或者恢復數據備份。 
有的數據庫在進行數據操做的時候,會生成一個「反操做」日誌。好比,insert 對 delete, 等。 

4. TransactionManager的實現可以處理多個XAResouce(一個XAResouce list)的狀況。 
好比Tyrex。或JBoss等EJB Server的Transaction實現代碼git

 

註釋:XA:XA協議由Tuxedo首先提出的,並交給X/Open組織,做爲資源管理器(數據庫)與事務管理器的接口標準。目前,Oracle、Informix、DB2和Sybase等各大數據庫廠家都提供對XA的支持。XA協議採用兩階段提交方式來管理分佈式事務。XA接口提供資源管理器與事務管理器之間進行通訊的標準接口。XA協議包括兩套函數,以xa_開頭的及以ax_開頭的。github

如下的函數使事務管理器能夠對資源管理器進行的操做:sql

  1)xa_open,xa_close:創建和關閉與資源管理器的鏈接。數據庫

  2)xa_start,xa_end:開始和結束一個本地事務。編程

  3)xa_prepare,xa_commit,xa_rollback:預提交、提交和回滾一個本地事務。框架

  4)xa_recover:回滾一個已進行預提交的事務。分佈式

  5)ax_開頭的函數使資源管理器能夠動態地在事務管理器中進行註冊,並能夠對XID(TRANSACTION IDS)進行操做。函數

  6)ax_reg,ax_unreg;容許一個資源管理器在一個TMS(TRANSACTION MANAGER SERVER)中動態註冊或撤消註冊。學習

 

 

JTA和JTS

 

JTA

 

JTA定義了一套接口,其中約定了幾種主要的角色:TransactionManager、UserTransaction、Transaction、XAResource,並定義了這些 角色之間須要遵照的規範,如Transaction的委託給TransactionManager等。

 

JTS

 

JTS也是一組規範,上面提到JTA中須要角色之間的交互,那應該如何交互?JTS就是約定了交互細節的規範。

 

整體上來講JTA更多的是從框架的角度來約定程序角色的接口,而JTS則是從具體實現的角度來約定程序角色之間的接口,二者各司其職。

 

由於JTA相對來講,更高層一些,咱們主要關注JTA。

 

XA

 

XA協議,規定事務管理器和資源管理器接口,採用二階段提交協議。

JTA事務有效的屏蔽了底層事務資源,可是與本地事務相比,XA協議的系統開銷大。

 

 

總結

 

關鍵接口

 

先介紹一下JTA的規範接口中,主要圍繞如下幾個接口:

 

  • UserTransaction:編程人員接口
  • TransactionManager:留給廠商實現的與事務管理有關的接口
  • Transaction:留給廠商實現的事務
  • XAResource:留給廠商實現的與持久化資源有關的接口

 

接口類中的接口定義可參見源碼,幾個接口類關係以下:

下載了JBoss的JTA實現源碼,可是發現深刻這個源碼有難度,且須要的時間也會較長,故留着之後學習。JBoss的JTA實現是Narayana,SVN地址爲https://github.com/jbosstm/narayana,若是有興趣能夠研究一下。

二階段提交協議

因此在一階段協議的基礎上,有了二階段協議,二階段協議的好處是添加了一個管理者角色,以下:

很明顯,二階段協議經過將兩層變爲三層,增長了中間的管理者角色,從而協調多個數據源之間的關係,二階段提交協議分爲兩個階段。

應用程序調用了事務管理器的提交方法,此後第一階段分爲兩個步驟:

  • 事務管理器通知參與該事務的各個資源管理器,通知他們開始準備事務。
  • 資源管理器接收到消息後開始準備階段,寫好事務日誌並執行事務,但不提交,而後將是否就緒的消息返回給事務管理器(此時已經將事務的大部分事情作完,之後的內容耗時極小)。

第二階段也分爲兩個步驟:    

  • 事務管理器在接受各個消息後,開始分析,若是有任意其一失敗,則發送回滾命令,不然發送提交命令。
  • 各個資源管理器接收到命令後,執行(耗時不多),並將提交消息返回給事務管理器。
    事務管理器接受消息後,事務結束,應用程序繼續執行。
    爲何要分兩步執行?一是由於分兩步,就有了事務管理器統一管理的機會;二儘量晚地提交事務,讓事務在提交前儘量地完成全部能完成的工做,這樣,最後的提交階段將是耗時極短,耗時極短意味着操做失敗的可能性也就下降。
    同時,二階段提交協議爲了保證事務的一致性,不論是事務管理器仍是各個資源管理器,每執行一步操做,都會記錄日誌,爲出現故障後的恢復準備依據。
    二階段提交協議的存在的弊端是阻塞,由於事務管理器要收集各個資源管理器的響應消息,若是其中一個或多個一直不返回消息,則事務管理器一直等待,應用程序也被阻塞,甚至可能永久阻塞。
相關文章
相關標籤/搜索