使用Spring中的 @Transactional註解控制事務
阿新 • • 發佈:2019-02-18
在Spring中有兩種事務,一種是宣告式事務,一種是程式設計式事務。
所謂的宣告式事務即為在配置檔案中配置,無需程式設計師手動程式設計控制事務,也就是說資料庫的事務的開啟,提交都是框架幫助我們做好的,
然而,程式設計式事務,是需要在方法中加入Spring的事務API 例如hibernate中的 beginTransaction() commit() rollback。。。。
顯然程式設計式事務更加具有細粒度,但是同時也增加了程式碼的傾入性。
現在筆者將以一個非常簡單的案例來描述@Transactional註解的使用。
注意,如果只是加了@Transactional註解,事務並不生效,@Transactional只是一種元資料而已。
並且,這裡有一點需要注意的是這個註解若是載入方法上,只能加在public的方法上,加到其他如private上不會報錯,但是事務控制就不會生效。
先上配置檔案
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx" xmlns="http://www.springframework.org/schema/beans" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <!--<bean id="myDate" class="java.util.Date"/>--> <!--<context:property-placeholder location="jdbc.properties"/>--> <!----> <!--<bean id="我是誰,我要到哪兒去,我為什麼這麼帥....." class="java.util.concurrent.CountDownLatch">--> <!--<constructor-arg value="2"/>--> <!--</bean>--> <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"> <property name="username" value="root"/> <property name="password" value=""/> <property name="url" value="jdbc:mysql://localhost:3306/test1"/> <property name="driverClassName" value="com.mysql.jdbc.Driver"/> </bean> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <property name="dataSource" ref="dataSource"/> </bean> <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource"/> </bean> <tx:annotation-driven transaction-manager="transactionManager" /> <context:component-scan base-package="springtransactionpackage"/> </beans>
配置檔案中配置了DataSource 和 JdbcTemplate以及 事務管理器。
<tx:annotation-driver transaction-manager="transactionManager">
這行程式碼是為了開啟 @Transactional 這個註解的使用
<context:component-scan base-package="springtransactionpackage">
這行程式碼是為了將發生事務的程式碼全部放入一個上下文中。
再來寫一個非常簡單的Dao
這裡我讓insertRecord()方法手動丟擲一個RuntimeException異常,使得整個事務失敗,package springtransactionpackage; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.stereotype.Repository; import org.springframework.transaction.annotation.Transactional; @Repository("myDao") public class TransactionAPITestDao { @Autowired private JdbcTemplate jdbcTemplate ; @Transactional(rollbackFor = {RuntimeException.class}) public void insertRecord() { System.out.println("開始插入資料"); jdbcTemplate.update( "insert into aaa values ('不好')" ); System.out.println("結束插入資料"); throw new RuntimeException( "Dao的時候手動丟擲異常" ); } }
package springtransactionpackage;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import javax.annotation.Resource;
import static org.junit.Assert.*;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:application.xml")
public class TransactionalAnnotationTest
{
@Resource
private TransactionAPITestDao transactionAPITestDao;
@Test
public void name() throws Exception
{
transactionAPITestDao.insertRecord();
}
}
最後再寫一個junit去測試一下,發現數據庫裡面並沒有出現事務失敗的資料。