Конфигурация двухфазной фиксации (2PC) с Atomikos
Я создаю пример приложения для тестирования двухфазной фиксации (2PC). Я взял биты кода, используемые здесь из Интернета. Я использую Spring, Hibernate и Atomikos с MySQL в качестве бэкэнда. Я использую две базы данных и намеренно совершаю вызов второй базы данных, чтобы не проверить, откатывается ли первый вызов базы данных. К сожалению, это не похоже на работу. Кто-нибудь может указать мне на некоторые ссылки с некоторым примером кода?
Следующее - моя конфигурация:
Фабрики Hibernate:
<bean id="sessionFactory1" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource">
<ref bean="dataSource1"/>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
<prop key="hibernate.show_sql">true</prop>
<prop key="hibernate.connection.isolation">3</prop>
<prop key="hibernate.current_session_context_class">jta</prop>
<prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JTATransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
<prop key="hibernate.connection.release_mode">on_close</prop>
</props>
</property>
<property name="mappingResources">
<list>
<value>/hibernate/Stock.hbm.xml</value>
</list>
</property>
</bean>
<bean id="sessionFactory2" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource">
<ref bean="dataSource2"/>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
<prop key="hibernate.show_sql">true</prop>
<prop key="hibernate.connection.isolation">3</prop>
<prop key="hibernate.current_session_context_class">jta</prop>
<prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JTATransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
<prop key="hibernate.connection.release_mode">on_close</prop>
</props>
</property>
<property name="mappingResources">
<list>
<value>/hibernate/Stock1.hbm.xml</value>
</list>
</property>
</bean>
Конфигурация источника данных:
<bean id="dataSource1" class="com.atomikos.jdbc.AtomikosDataSourceBean" init-method="init" destroy-method="close">
<!-- set an arbitrary but unique name for the datasource -->
<property name="uniqueResourceName"><value>XADBMS1</value></property>
<!-- set the underlying driver class to use, in this example case we use MySql -->
<property name="xaDataSourceClassName">
<value>com.mysql.jdbc.jdbc2.optional.MysqlXADataSource</value>
</property>
<property name="xaProperties">
<!-- set the driver-specific XADataSource properties (check your driver docs for more info) -->
<props>
<prop key="user">${jdbc.username}</prop>
<prop key="password">${jdbc.password}</prop>
<prop key="URL" >${jdbc.url1}</prop>
</props>
</property>
<!-- how many connections in the pool? -->
<property name="poolSize" value="3"/>
</bean>
<bean id="dataSource2" class="com.atomikos.jdbc.AtomikosDataSourceBean" init-method="init" destroy-method="close">
<!-- set an arbitrary but unique name for the datasource -->
<property name="uniqueResourceName"><value>XADBMS2</value></property>
<!-- set the underlying driver class to use, in this example case we use MySql -->
<property name="xaDataSourceClassName">
<value>com.mysql.jdbc.jdbc2.optional.MysqlXADataSource</value>
</property>
<property name="xaProperties">
<!-- set the driver-specific XADataSource properties (check your driver docs for more info) -->
<props>
<prop key="user">${jdbc.username}</prop>
<prop key="password">${jdbc.password}</prop>
<prop key="URL" >${jdbc.url2}</prop>
</props>
</property>
<!-- how many connections in the pool? -->
<property name="poolSize" value="3"/>
</bean>
Конфигурация Spring JTA:
<bean id="AtomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close">
<!-- when close is called, should we force transactions to terminate or not? -->
<property name="forceShutdown" value="false" />
</bean>
<!--Also use Atomikos UserTransactionImp, needed to configure Spring -->
<bean id="AtomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
<property name="transactionTimeout" value="300" />
</bean>
<!--Configure the Spring framework to use JTA transactions from Atomikos -->
<bean id="JtaTransactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManager" ref="AtomikosTransactionManager" />
<property name="userTransaction" ref="AtomikosUserTransaction" />
</bean>
У меня есть два DAOImpl, в которые я вставляю две фабрики сессии, определенные выше. Ниже приведен вызов из кода Java:
public static void main( String[] args )
{
ApplicationContext appContext =
new ClassPathXmlApplicationContext("spring/config/appContext.xml");
StockBo stockBo = (StockBo)appContext.getBean("stockBo1");
StockBo stockBo2 = (StockBo)appContext.getBean("stockBo2");
/** insert **/
Stock stock = new Stock();
stock.setStockCode("7668");
stock.setStockName("HAIO");
stockBo.save(stock);
Stock stock1 = new Stock();
//stock1.setStockCode("1668"); **Commented to fail the second db insert**
stock1.setStockName("AAIO");
stockBo2.save(stock1);
}
Любые указатели будут очень полезны.
Спасибо
1 ответ
Чтобы получить ожидаемое поведение, необходимо выполнить вызовы DAO внутри одной транзакции, например, следующим образом:
final StockBo stockBo = (StockBo)appContext.getBean("stockBo1");
final StockBo stockBo2 = (StockBo)appContext.getBean("stockBo2");
TransactionTemplate tx = new TransactionTemplate(appContext.getBean(PlatformTransactionManager.class);
tx.execute(new TransactionCallback<Void>() {
public Void doInTransaction(TransactionStatus ts) {
/** insert **/
Stock stock = new Stock();
stock.setStockCode("7668");
stock.setStockName("HAIO");
stockBo.save(stock);
Stock stock1 = new Stock();
//stock1.setStockCode("1668"); **Commented to fail the second db insert**
stock1.setStockName("AAIO");
stockBo2.save(stock1);
return null;
}
});
Кроме того, я думаю, что вам не нужны следующие строки, так как вы настраиваете Hibernate с помощью Spring:
<prop key="hibernate.current_session_context_class">jta</prop>
<prop key="hibernate.transaction.factory_class">org.hibernate.transaction.JTATransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
<prop key="hibernate.connection.release_mode">on_close</prop>
Смотрите также: