您好,登錄后才能下訂單哦!
這篇文章給大家介紹Spring中怎么實現聲明式事務,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
Spring事務管理
Spring支持兩種方式的事務管理:
編程式事務管理: 通過Transaction Template手動管理事務,實際應用中很少使用,
使用XML配置聲明式事務: 推薦使用(代碼侵入性最小),實際是通過AOP實現
實現聲明式事務的四種方式:
基于 TransactionInterceptor 的聲明式事務: Spring 聲明式事務的基礎,通常也不建議使用這種方式,但是與前面一樣,了解這種方式對理解 Spring 聲明式事務有很大作用。
基于 TransactionProxyFactoryBean 的聲明式事務: 第一種方式的改進版本,簡化的配置文件的書寫,這是 Spring 早期推薦的聲明式事務管理方式,但是在 Spring 2.0 中已經不推薦了。
基于< tx> 和< aop>命名空間的聲明式事務管理: 目前推薦的方式,其最大特點是與 Spring AOP 結合緊密,可以充分利用切點表達式的強大支持,使得管理事務更加靈活。
基于 @Transactional 的全注解方式: 將聲明式事務管理簡化到了極致。開發人員只需在配置文件中加上一行啟用相關后處理 Bean 的配置,然后在需要實施事務管理的方法或者類上使用 @Transactional 指定事務規則即可實現事務管理,而且功能也不必其他方式遜色。
我們今天要將的是使用編程式以及基于AspectJ的聲明式和基于注解的事務方式,實現爛大街的轉賬業務。
再來說一下這個案例的思想吧,我們在兩次轉賬之間添加一個錯誤語句(對應銀行斷電等意外情況),如果這個時候兩次轉賬不能成功,則說明事務配置正確,否則,事務配置不正確。
你需要完成的任務:
使用編程式事務管理完成轉賬業務
使用基于AspectJ的聲明式事務管理完成轉賬業務
使用基于 @Transactional 的全注解方式事務管理完成轉賬業務
備注:
下面的代碼是在很久之前,我剛學Sping還沒有接觸Maven的時候寫的,所以我使用的原始添加jar的方式,使用Maven的小伙伴可以自行添加Maven依賴
項目結構:
Spring編程式和聲明式事務實例講解
開發工具:
Myeclipse2017
SQL:
create table `account` ( `username` varchar (99), `salary` int (11) ); insert into `account` (`username`, `salary`) values('小王','3000'); insert into `account` (`username`, `salary`) values('小馬','3000');
(1)編程式事務管理
注意: 通過添加/刪除accountMoney() 方法中int i = 10 / 0這個語句便可驗證事務管理是否配置正確。
OrdersDao.java(Dao層)
package cn.itcast.dao; import org.springframework.jdbc.core.JdbcTemplate; public class OrdersDao { // 注入jdbcTemplate模板對象 private JdbcTemplate jdbcTemplate; public void setJdbcTemplate(JdbcTemplate jdbcTemplate) { this.jdbcTemplate = jdbcTemplate; } // 對數據操作的方法不包含業務操作 /** * 小王少錢的方法 */ public void reduceMoney() { String sql = "update account set salary=salary-? where username=?"; jdbcTemplate.update(sql, 1000, "小王"); } /** * 小馬多錢的方法 */ public void addMoney() { String sql = "update account set salary=salary+? where username=?"; jdbcTemplate.update(sql, 1000, "小馬"); } }
OrdersService.java(業務邏輯層)
package cn.itcast.service; import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.support.TransactionCallback; import org.springframework.transaction.support.TransactionTemplate; import cn.itcast.dao.OrdersDao; public class OrdersService { // 注入Dao層對象 private OrdersDao ordersDao; public void setOrdersDao(OrdersDao ordersDao) { this.ordersDao = ordersDao; } // 注入TransactionTemplate對象 private TransactionTemplate transactionTemplate; public void setTransactionTemplate(TransactionTemplate transactionTemplate) { this.transactionTemplate = transactionTemplate; } // 調用dao的方法 // 業務邏輯,寫轉賬業務 public void accountMoney() { transactionTemplate.execute(new TransactionCallback<Object>() { @Override public Object doInTransaction(TransactionStatus status) { Object result = null; try { // 小馬多1000 ordersDao.addMoney(); // 加入出現異常如下面int // i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢 // 解決辦法是出現異常后進行事務回滾 int i = 10 / 0;// 事務管理配置后異常已經解決 // 小王 少1000 ordersDao.reduceMoney(); } catch (Exception e) { status.setRollbackOnly(); result = false; System.out.println("Transfer Error!"); } return result; } }); } }
TestService.java(測試方法)
package cn.itcast.service; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestService { @Test public void testAdd() { ApplicationContext context = new ClassPathXmlApplicationContext( "beans.xml"); OrdersService userService = (OrdersService) context .getBean("ordersService"); userService.accountMoney(); } }
配置文件:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd"> <!-- 配置c3po連接池 --> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <!-- 注入屬性值 --> <property name="driverClass" value="com.mysql.jdbc.Driver"></property> <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property> <property name="user" value="root"></property> <property name="password" value="153963"></property> </bean> <!-- 編程式事務管理 --> <!-- 配置事務管理器 --> <bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <!-- 注入dataSource --> <property name="dataSource" ref="dataSource"></property> </bean> <!-- 配置事務管理器模板 --> <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate"> <!-- 注入真正進行事務管理的事務管理器,name必須為 transactionManager否則無法注入 --> <property name="transactionManager" ref="dataSourceTransactionManager"></property> </bean> <!-- 對象生成及屬性注入 --> <bean id="ordersService" class="cn.itcast.service.OrdersService"> <property name="ordersDao" ref="ordersDao"></property> <!-- 注入事務管理的模板 --> <property name="transactionTemplate" ref="transactionTemplate"></property> </bean> <bean id="ordersDao" class="cn.itcast.dao.OrdersDao"> <property name="jdbcTemplate" ref="jdbcTemplate"></property> </bean> <!-- JDBC模板對象 --> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <property name="dataSource" ref="dataSource"></property> </bean> </beans>
(2)基于AspectJ的聲明式事務管理
OrdersService.java(業務邏輯層)
package cn.itcast.service; import cn.itcast.dao.OrdersDao; public class OrdersService { private OrdersDao ordersDao; public void setOrdersDao(OrdersDao ordersDao) { this.ordersDao = ordersDao; } // 調用dao的方法 // 業務邏輯,寫轉賬業務 public void accountMoney() { // 小馬多1000 ordersDao.addMoney(); // 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢 // 解決辦法是出現異常后進行事務回滾 int i = 10 / 0;// 事務管理配置后異常已經解決 // 小王 少1000 ordersDao.reduceMoney(); } }
配置文件:
<!-- 配置c3po連接池 --> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <!-- 注入屬性值 --> <property name="driverClass" value="com.mysql.jdbc.Driver"></property> <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property> <property name="user" value="root"></property> <property name="password" value="153963"></property> </bean> <!-- 第一步:配置事務管理器 --> <bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <!-- 注入dataSource --> <property name="dataSource" ref="dataSource"></property> </bean> <!-- 第二步:配置事務增強 --> <tx:advice id="txadvice" transaction-manager="dataSourceTransactionManager"> <!-- 做事務操作 --> <tx:attributes> <!-- 設置進行事務操作的方法匹配規則 --> <!-- account開頭的所有方法 --> <!-- propagation:事務傳播行為; isolation:事務隔離級別; read-only:是否只讀; rollback-for:發生那些異常時回滾 timeout:事務過期時間 --> <tx:method name="account*" propagation="REQUIRED" isolation="DEFAULT" read-only="false" rollback-for="" timeout="-1" /> </tx:attributes> </tx:advice> <!-- 第三步:配置切面 切面即把增強用在方法的過程 --> <aop:config> <!-- 切入點 --> <aop:pointcut expression="execution(* cn.itcast.service.OrdersService.*(..))" id="pointcut1" /> <!-- 切面 --> <aop:advisor advice-ref="txadvice" pointcut-ref="pointcut1" /> </aop:config> <!-- 對象生成及屬性注入 --> <bean id="ordersService" class="cn.itcast.service.OrdersService"> <property name="ordersDao" ref="ordersDao"></property> </bean> <bean id="ordersDao" class="cn.itcast.dao.OrdersDao"> <property name="jdbcTemplate" ref="jdbcTemplate"></property> </bean> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <property name="dataSource" ref="dataSource"></property> </bean>
(3)基于注解的方式
OrdersService.java(業務邏輯層)
package cn.itcast.service; import org.springframework.transaction.annotation.Isolation; import org.springframework.transaction.annotation.Propagation; import org.springframework.transaction.annotation.Transactional; import cn.itcast.dao.OrdersDao; @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false, timeout = -1) public class OrdersService { private OrdersDao ordersDao; public void setOrdersDao(OrdersDao ordersDao) { this.ordersDao = ordersDao; } // 調用dao的方法 // 業務邏輯,寫轉賬業務 public void accountMoney() { // 小馬多1000 ordersDao.addMoney(); // 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢 // 解決辦法是出現異常后進行事務回滾 // int i = 10 / 0;// 事務管理配置后異常已經解決 // 小王 少1000 ordersDao.reduceMoney(); } }
配置文件:
<!-- 配置c3po連接池 --> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <!-- 注入屬性值 --> <property name="driverClass" value="com.mysql.jdbc.Driver"></property> <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property> <property name="user" value="root"></property> <property name="password" value="153963"></property> </bean> <!-- 第一步:配置事務管理器 (和配置文件方式一樣)--> <bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <!-- 注入dataSource --> <property name="dataSource" ref="dataSource"></property> </bean> <!-- 第二步: 開啟事務注解 --> <tx:annotation-driven transaction-manager="dataSourceTransactionManager" /> <!-- 第三步 在方法所在類上加注解 --> <!-- 對象生成及屬性注入 --> <bean id="ordersService" class="cn.itcast.service.OrdersService"> <property name="ordersDao" ref="ordersDao"></property> </bean> <bean id="ordersDao" class="cn.itcast.dao.OrdersDao"> <property name="jdbcTemplate" ref="jdbcTemplate"></property> </bean> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <property name="dataSource" ref="dataSource"></property> </bean>
關于Spring中怎么實現聲明式事務就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。