Spring分布式事務實現
分布式事務是指操作多個數據庫之間的事務,spring的org.springframework.transaction.jta.JtaTransactionManager,提供了分布式事務支持。如果使用WAS的JTA支持,把它的屬性改為WebSphere對應的TransactionManager。
在tomcat下,是沒有分布式事務的,不過可以借助于第三方軟件jotm(Java Open Transaction Manager )和AtomikosTransactionsEssentials實現,在spring中分布式事務是通過jta(jotm,atomikos)來進行實現。
1、http://jotm.objectweb.org/
2、http://www.atomikos.com/Main/TransactionsEssentials
一、使用JOTM例子
(1) Dao及實現
GenericDao接口:
public interface GenericDao { public int save(String ds, String sql, Object[] obj) throws Exception; public int findRowCount(String ds, String sql); }
GenericDaoImpl 實現:
public class GenericDaoImpl implements GenericDao{private JdbcTemplate jdbcTemplateA; private JdbcTemplate jdbcTemplateB; public void setJdbcTemplateA(JdbcTemplate jdbcTemplate) { this.jdbcTemplateA = jdbcTemplate; } public void setJdbcTemplateB(JdbcTemplate jdbcTemplate) { this.jdbcTemplateB = jdbcTemplate; } public int save(String ds, String sql, Object[] obj) throws Exception{ if(null == ds || "".equals(ds)) return -1; try{ if(ds.equals("A")){ return this.jdbcTemplateA.update(sql, obj); }else{ return this.jdbcTemplateB.update(sql, obj); } }catch(Exception e){ e.printStackTrace(); throw new Exception("執行" + ds + "數據庫時失敗!"); } } public int findRowCount(String ds, String sql) { if(null == ds || "".equals(ds)) return -1; if(ds.equals("A")){ return this.jdbcTemplateA.queryForInt(sql); }else{ return this.jdbcTemplateB.queryForInt(sql); } }
}</pre>
(2) Service及實現
UserService 接口:
public interface UserService { public void saveUser() throws Exception; }UserServiceImpl 實現:
public class UserServiceImpl implements UserService{private GenericDao genericDao; public void setGenericDao(GenericDao genericDao) { this.genericDao = genericDao; } public void saveUser() throws Exception { String userName = "user_" + Math.round(Math.random()*10000); System.out.println(userName); StringBuilder sql = new StringBuilder(); sql.append(" insert into t_user(username, gender) values(?,?); "); Object[] objs = new Object[]{userName,"1"}; genericDao.save("A", sql.toString(), objs); sql.delete(0, sql.length()); sql.append(" insert into t_user(name, sex) values(?,?); "); objs = new Object[]{userName,"男的"};//值超出范圍 genericDao.save("B", sql.toString(), objs); }
}</pre>
(3) applicationContext-jotm.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="</beans></pre>
(4) 測試
public class TestUserService{private static UserService userService; @BeforeClass public static void init(){ ApplicationContext app = new ClassPathXmlApplicationContext("applicationContext-jotm.xml"); userService = (UserService)app.getBean("userService"); } @Test public void save(){ System.out.println("begin..."); try{ userService.saveUser(); }catch(Exception e){ System.out.println(e.getMessage()); } System.out.println("finish..."); }
}</pre>
二、關于使用atomikos實現
(1) 數據源配置<bean id="dataSourceA" class="com.atomikos.jdbc.SimpleDataSourceBean" init-method="init" destroy-method="close"> <property name="uniqueResourceName"> <value>${datasource.uniqueResourceName}</value> </property> <property name="xaDataSourceClassName"> <value>${database.driver_class}</value> </property> <property name="xaDataSourceProperties"> <value>URL=${database.url};user=${database.username};password=${database.password}</value> </property> <property name="exclusiveConnectionMode"> <value>${connection.exclusive.mode}</value> </property> <property name="connectionPoolSize"> <value>${connection.pool.size}</value> </property> <property name="connectionTimeout"> <value>${connection.timeout}</value> </property> <property name="validatingQuery"> <value>SELECT 1</value> </property> </bean>
(2)、事務配置
<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close"> <property name="forceShutdown" value="true"/> </bean><bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp"> <property name="transactionTimeout" value="${transaction.timeout}"/> </bean> <!-- JTA事務管理器 --> <bean id="springTransactionManager" class="org.springframework.transaction.jta.JtaTransactionManager"> <property name="transactionManager" ref="atomikosTransactionManager"/> <property name="userTransaction" ref="atomikosUserTransaction"/> </bean> <!-- 事務切面配置 --> <aop:config> <aop:pointcut id="serviceOperation" expression="execution(* *..service*..*(..))"/> <aop:advisor pointcut-ref="serviceOperation" advice-ref="txAdvice"/> </aop:config> <!-- 通知配置 --> <tx:advice id="txAdvice" transaction-manager="springTransactionManager"> <tx:attributes> <tx:method name="*" rollback-for="Exception"/> </tx:attributes> </tx:advice> </pre><br />
有關JTA
JTA全稱為Java Transaction API,顧名思義JTA定義了一組統一的事務編程的接口,這些接口如下:
XAResource
XAResource接口是對實現了X/Open CAE規范的資源管理器 (Resource Manager,數據庫就是典型的資源管理器) 的抽象,它由資源適配器 (Resource Apdater) 提供實現。XAResource是支持事務控制的核心。
Transaction
Transaction接口是一個事務實例的抽象,通過它可以控制事務內多個資源的提交或者回滾。二階段提交過程也是由Transaction接口的實現者來完成的。TransactionManager
托管模式 (managed mode) 下,TransactionManager接口是被應用服務器調用,以控制事務的邊界的。
UserTransaction
非托管模式 (non-managed mode) 下,應用程序可以通過UserTransaction接口控制事務的邊界
托管模式下的事務提交場景
注意:在上圖中3和5的步驟之間省略了應用程序對資源的操作 (如CRUD)。另外,應用服務器什么時機 enlistResource,又是什么時候delistResource呢?這在后文中會解釋。
有關JCA
下圖為JCA的架構圖
中間涉及元素說明如下:
1)Enterprise Information System
簡稱EIS,在JTA中它又被稱為資源管理器。典型的EIS有數據庫,事務處理系統(Transaction Processing System),ERP系統。
2)Resource Adapter
資源適配器(Resource Adaper)是JCA的關鍵。要想把不同的EIS整合(或者連接)到J2EE運行環境中,就必須為每個EIS提供資源適配器,它會將將EIS適配為一個具備統一編程接口的資源 (Resource) 。這個統一編程接口就是上圖中的System Contracts和Client API。下面的UML類圖將完美詮釋資源適配器。
3)Application Server
應用服務器 (Application Server) 通過System Contracts來管理對EIS的安全、事務、連接等。典型的應用服務器有JBoss、JOnAS、Geronimo、GlassFish等。
4)Application Component
應用組件 (Application Component) ,它封裝了應用業務邏輯,像對資源的訪問和修改。典型的應用組件就是EJB。
更多細節請參見:
Sun Microsystems Inc.J2EE Connector Architecture 1.5來自:http://blog.csdn.net/ithomer/article/details/10859235