您现在的位置是:首页 > 文章详情

Spring编程式和声明式事务实例讲解

日期:2018-05-22点击:258

Java面试通关手册(Java学习指南):https://github.com/Snailclimb/Java_Guide

历史回顾:
可能是最漂亮的Spring事务管理详解

Spring事务管理

Spring支持两种方式的事务管理:

  • 编程式事务管理: 通过Transaction Template手动管理事务,实际应用中很少使用,
  • 使用XML配置声明式事务: 推荐使用(代码侵入性最小),实际是通过AOP实现

实现声明式事务的四种方式:

  1. 基于 TransactionInterceptor 的声明式事务: Spring 声明式事务的基础,通常也不建议使用这种方式,但是与前面一样,了解这种方式对理解 Spring 声明式事务有很大作用。
  2. 基于 TransactionProxyFactoryBean 的声明式事务: 第一种方式的改进版本,简化的配置文件的书写,这是 Spring 早期推荐的声明式事务管理方式,但是在 Spring 2.0 中已经不推荐了。
  3. 基于< tx> 和< aop>命名空间的声明式事务管理: 目前推荐的方式,其最大特点是与 Spring AOP 结合紧密,可以充分利用切点表达式的强大支持,使得管理事务更加灵活。
  4. 基于 @Transactional 的全注解方式: 将声明式事务管理简化到了极致。开发人员只需在配置文件中加上一行启用相关后处理 Bean 的配置,然后在需要实施事务管理的方法或者类上使用 @Transactional 指定事务规则即可实现事务管理,而且功能也不必其他方式逊色。

我们今天要将的是使用编程式以及基于AspectJ的声明式和基于注解的事务方式,实现烂大街的转账业务。

再来说一下这个案例的思想吧,我们在两次转账之间添加一个错误语句(对应银行断电等意外情况),如果这个时候两次转账不能成功,则说明事务配置正确,否则,事务配置不正确。

你需要完成的任务:

  • 使用编程式事务管理完成转账业务
  • 使用基于AspectJ的声明式事务管理完成转账业务
  • 使用基于 @Transactional 的全注解方式事务管理完成转账业务

备注:

下面的代码是在很久之前,我刚学Sping还没有接触Maven的时候写的,所以我使用的原始添加jar的方式,使用Maven的小伙伴可以自行添加Maven依赖,没有使用Maven的小伙伴直接使用我下面提供的jar包即可。

jar包地址:链接:https://pan.baidu.com/s/1tqy-mVKxSutsIIvYgtC3Rw 密码:nid0

项目结构:
项目结构

开发工具:

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>

欢迎关注我的微信公众号: "Java面试通关手册" (一个有温度的微信公众号,期待与你共同进步~~~坚持原创,分享美文,分享各种Java学习资源):

原文链接:https://yq.aliyun.com/articles/596172
关注公众号

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。

持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。

转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。

文章评论

共有0条评论来说两句吧...

文章二维码

扫描即可查看该文章

点击排行

推荐阅读

最新文章