Spring中事务管理的四种方法(银行转账为例)

 更新时间:2018年05月15日 11:33:15   作者:皮卡丘啾啾  
这篇文章主要给大家介绍了关于Spring中事务管理的四种方法,文中是以银行转账为例,通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

本文配套示例代码下载地址(完整可运行,含sql文件,下载后请修改数据库配置):点击这里下载

一、事务的作用

  将若干的数据库操作作为一个整体控制,一起成功或一起失败。

  原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。

  一致性:指事务前后数据的完整性必须保持一致。

  隔离性:指多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离。

  持久性:指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,即时数据库发生故障也不应该对其有任何影响。

二、Spring事务管理高层抽象主要包括3个接口

  --Platform TransactionManager 事务管理器(提交、回滚事务)

     Spring为不同的持久化框架提供了不同的Platform TransactionManager接口实现。如:

        使用Spring JDBC或iBatis进行持久化数据时使用DataSourceTransactionManager

        使用Hibernate3.0版本进行持久化数据时使用HibernateTransactionManager

  --TransactionDefinition 事务定义信息(隔离、传播、超时、只读)

        脏读:一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。

        不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同。

        幻读:一个事务读取了几行记录后,另一个事务插入一些记录,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。

        事务隔离级别:(五种)

  •     DEFAULT--使用后端数据库默认的隔离级别(Spring中的选择项)
  •     READ_UNCOMMITED--允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读
  •     READ_COMMITTED--允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生
  •     REPEATABLE_READ--对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生
  •     SERIALIZABLE--完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的

    其中,MySQL默认采用REPEATABLE_READ隔离级别;Oracle默认采用READ_COMMITTED隔离级别

        事务传播行为:(七种)

  •     REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
  •     SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
  •     MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
  •     REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
  •     NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  •     NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
  •     NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与REQUIRED类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对DataSourceTransactionManager有效

  --TransactionStatus 事务具体运行状态

三、Spring提供了以下方法控制事务

  a.编程式事务管理(基于Java编程控制,很少使用)--见demo1包

       利用TransactionTemplate将多个DAO操作封装起来

  *b.声明式事务管理(基于Spring的AOP配置控制)

       -基于TransactionProxyFactoryBean的方式.(很少使用)--见demo2包

            需要为每个进行事务管理的类,配置一个TransactionProxyFactoryBean进行增强.

       -基于XML配置(经常使用)--见demo3包

            一旦配置好之后,类上不需要添加任何东西。

            如果Action作为目标对象切入事务,需要在<aop:config>元素里添加proxy-target-class="true"属性。原因是通知Spring框架采用CGLIB技术生成具有事务管理功能的Action类。

       -基于注解(配置简单,经常使用)--见demo4包

            在applicationContext.xml中开启事务注解配置。(applicationContext.xml中只需定义Bean并追加以下元素)

<bean id="txManager" class="...">
 <property name="sessionFactory">
 </property>
<tx:annotation-driven transaction-manager="txManager"/>

            在目标组件类中使用@Transactional,该标记可定义在类前或方法前。

四、示例(银行转账)

        --编程式

/** 
 * @Description:转账案例的DAO层接口 
 * 
 */ 
public interface AccountDao { 
 /** 
 * @param out 
 * :转出账号 
 * @param money 
 * :转账金额 
 */ 
 public void outMoney(String out, Double money); 
 
 /** 
 * 
 * @param in 
 * :转入账号 
 * @param money 
 * :转账金额 
 */ 
 public void inMoney(String in, Double money); 
} 
/** 
 * @Description:转账案例的DAO层实现类 
 */ 
public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao { 
 /** 
 * @param out 
 * :转出账号 
 * @param money 
 * :转账金额 
 */ 
 @Override 
 public void outMoney(String out, Double money) { 
 String sql = "update account set money = money-? where name = ?"; 
 this.getJdbcTemplate().update(sql, money, out); 
 } 
 /** 
 * @param in 
 * :转入账号 
 * @param money 
 * :转账金额 
 */ 
 @Override 
 public void inMoney(String in, Double money) { 
 String sql = "update account set money = money+? where name = ?"; 
 this.getJdbcTemplate().update(sql, money, in); 
 } 
} 
/** 
 * @Description:转账案例的业务接口 
 * 
 */ 
public interface AccountService { 
 /** 
 * @param out :转出账号 
 * @param in :转入账号 
 * @param money :转账金额 
 */ 
 public void transfer(String out,String in,Double money); 
} 
/** 
 * @Description:转账案例的业务层实现类 
 */ 
public class AccountServiceImpl implements AccountService { 
 // 注入转账的DAO 
 private AccountDao accountDao; 
 
 // 注入事务管理的模板 
 private TransactionTemplate transactionTemplate; 
 
 /** 
 * @param out 
 * :转出账号 
 * @param in 
 * :转入账号 
 * @param money 
 * :转账金额 
 */ 
 @Override 
 public void transfer(final String out, final String in, final Double money) { 
 
 // 未经事务控制的业务处理操作,如果过程中出异常,则导致前面的操作能完成,后面的不能,即转账成功但未收到转账款 
 // accountDao.outMoney(out, money); 
 // int i = 1/0; 
 // accountDao.inMoney(in, money); 
 
 transactionTemplate.execute(new TransactionCallbackWithoutResult() { 
 
 @Override 
 protected void doInTransactionWithoutResult( 
  TransactionStatus transactionStatus) { 
 accountDao.outMoney(out, money); 
 // int i = 1 / 0;//事务控制,即出现异常,该段内代码都执行失效 
 accountDao.inMoney(in, money); 
 } 
 }); 
 } 
 
 public void setAccountDao(AccountDao accountDao) { 
 this.accountDao = accountDao; 
 } 
 
 public void setTransactionTemplate(TransactionTemplate transactionTemplate) { 
 this.transactionTemplate = transactionTemplate; 
 } 
} 

applicationContext1.xml

<!-- 引入外部的属性文件 --> 
 <context:property-placeholder location="classpath:jdbc.properties"/> 
 
 <!-- 配置c3p0连接池 --> 
 <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> 
 <property name="driverClass" value="${jdbc.driverClass}" /> 
 <property name="jdbcUrl" value="${jdbc.url}" /> 
 <property name="user" value="${jdbc.username}" /> 
 <property name="password" value="${jdbc.password}" /> 
 </bean> 
 
 <!-- 配置业务层类 --> 
 <bean id="accountService" class="com.zs.spring.demo1.AccountServiceImpl"> 
 <property name="accountDao" ref="accountDao" /> 
 <!-- 注入事务管理的模板 --> 
 <property name="transactionTemplate" ref="transactionTemplate" /> 
 </bean> 
 
 <!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> 
 <bean id="accountDao" class="com.zs.spring.demo1.AccountDaoImpl"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- 配置DAO类(未简化) --> 
 <!-- <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 <bean id="accountDao" class="com.zs.spring.demo1.AccountDaoImpl"> 
 <property name="jdbcTemplate" ref="jdbcTemplate" /> 
 </bean> --> 
 
 <!-- ==================================1.编程式的事务管理=============================================== --> 
 <!-- 配置事务管理器 --> 
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- 配置事务管理的模板:Spring为了简化事务管理的代码而提供的类 --> 
 <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate"> 
 <property name="transactionManager" ref="transactionManager"/> 
 </bean> 

测试:

@RunWith(SpringJUnit4ClassRunner.class) 
@ContextConfiguration("classpath:applicationContext1.xml") 
public class TransactionTest { 
 @Resource(name = "accountService") 
 private AccountService accountService; 
 
 @Test 
 public void demo1() { 
 accountService.transfer("aaa", "bbb", 200d); 
 } 
} 

    --基于TransactionProxyFactoryBean的方式

public class AccountServiceImpl implements AccountService { 
 // 注入转账的DAO 
 private AccountDao accountDao; 
 
 /** 
 * @param out 
 *  :转出账号 
 * @param in 
 *  :转入账号 
 * @param money 
 *  :转账金额 
 */ 
 @Override 
 public void transfer(String out, String in, Double money) { 
 accountDao.outMoney(out, money); 
 // int i = 1/0; 
 accountDao.inMoney(in, money); 
 } 
 
 public void setAccountDao(AccountDao accountDao) { 
 this.accountDao = accountDao; 
 } 
} 

applicationContext2.xml

<!-- 引入外部的属性文件 --> 
 <context:property-placeholder location="classpath:jdbc.properties"/> 
 
 <!-- 配置c3p0连接池 --> 
 <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> 
 <property name="driverClass" value="${jdbc.driverClass}" /> 
 <property name="jdbcUrl" value="${jdbc.url}" /> 
 <property name="user" value="${jdbc.username}" /> 
 <property name="password" value="${jdbc.password}" /> 
 </bean> 
 
 <!-- 配置业务层类 --> 
 <bean id="accountService" class="com.zs.spring.demo2.AccountServiceImpl"> 
 <property name="accountDao" ref="accountDao" /> 
 </bean> 
 
 <!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> 
 <bean id="accountDao" class="com.zs.spring.demo2.AccountDaoImpl"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- ==================================2.使用XML配置声明式的事务管理(原始方式)=============================================== --> 
 
 <!-- 配置事务管理器 --> 
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- 配置业务层的代理 --> 
 <bean id="accountServiceProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> 
 <!-- 配置目标对象 --> 
 <property name="target" ref="accountService" /> 
 <!-- 注入事务管理器 --> 
 <property name="transactionManager" ref="transactionManager"></property> 
 <!-- 注入事务的属性 --> 
 <property name="transactionAttributes"> 
  <props> 
  <!-- 
   prop的格式: 
   * PROPAGATION :事务的传播行为 
   * ISOTATION :事务的隔离级别 
   * readOnly :只读 
   * -EXCEPTION :发生哪些异常回滚事务 
   * +EXCEPTION :发生哪些异常不回滚事务 
   --> 
  <prop key="transfer">PROPAGATION_REQUIRED</prop> 
  <!-- <prop key="transfer">PROPAGATION_REQUIRED,readOnly</prop> --> 
  <!-- <prop key="transfer">PROPAGATION_REQUIRED,+java.lang.ArithmeticException</prop> --> 
  </props> 
 </property> 
 </bean> 

测试:

@RunWith(SpringJUnit4ClassRunner.class) 
@ContextConfiguration("classpath:applicationContext2.xml") 
public class TransactionTest { 
 /** 
 * 一定要注入代理类:因为代理类进行增强的操作 
 */ 
 // @Resource(name="accountService") 
 @Resource(name = "accountServiceProxy") 
 private AccountService accountService; 
 
 @Test 
 public void demo1() { 
 accountService.transfer("aaa", "bbb", 200d); 
 } 
} 

    --基于XML配置

public class AccountServiceImpl implements AccountService { 
 // 注入转账的DAO 
 private AccountDao accountDao; 
 
 /** 
 * @param out 
 *  :转出账号 
 * @param in 
 *  :转入账号 
 * @param money 
 *  :转账金额 
 */ 
 @Override 
 public void transfer(String out, String in, Double money) { 
 accountDao.outMoney(out, money); 
 // int i = 1/0; 
 accountDao.inMoney(in, money); 
 
 } 
 
 public void setAccountDao(AccountDao accountDao) { 
 this.accountDao = accountDao; 
 } 
} 

applicationContext3.xml

<!-- 引入外部的属性文件 --> 
 <context:property-placeholder location="classpath:jdbc.properties"/> 
 
 <!-- 配置c3p0连接池 --> 
 <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> 
 <property name="driverClass" value="${jdbc.driverClass}" /> 
 <property name="jdbcUrl" value="${jdbc.url}" /> 
 <property name="user" value="${jdbc.username}" /> 
 <property name="password" value="${jdbc.password}" /> 
 </bean> 
 
 <!-- 配置业务层类 --> 
 <bean id="accountService" class="com.zs.spring.demo3.AccountServiceImpl"> 
 <property name="accountDao" ref="accountDao" /> 
 </bean> 
 
 <!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> 
 <bean id="accountDao" class="com.zs.spring.demo3.AccountDaoImpl"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- ==================================3.使用XML配置声明式的事务管理,基于tx/aop=============================================== --> 
 
 <!-- 配置事务管理器 --> 
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- 配置事务的通知 --> 
 <tx:advice id="txAdvice" transaction-manager="transactionManager"> 
 <tx:attributes> 
  <!-- 
  propagation :事务传播行为 
  isolation :事务的隔离级别 
  read-only :只读 
  rollback-for:发生哪些异常回滚 
  no-rollback-for :发生哪些异常不回滚 
  timeout :过期信息 
  --> 
  <tx:method name="transfer" propagation="REQUIRED"/> 
 </tx:attributes> 
 </tx:advice> 
 
 <!-- 配置切面 --> 
 <aop:config> 
 <!-- 配置切入点 --> 
 <aop:pointcut expression="execution(* com.zs.spring.demo3.AccountService+.*(..))" id="pointcut1"/> 
 <!-- 配置切面 --> 
 <aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1"/> 
 </aop:config> 

测试:

/** 
 * @Description:Spring的声明式事务管理的方式二:基于AspectJ的XML方式的配置 
 */ 
@RunWith(SpringJUnit4ClassRunner.class) 
@ContextConfiguration("classpath:applicationContext3.xml") 
public class TransactionTest { 
 /** 
 * 一定要注入代理类:因为代理类进行增强的操作 
 */ 
 @Resource(name = "accountService") 
 private AccountService accountService; 
 
 @Test 
 public void demo1() { 
 accountService.transfer("aaa", "bbb", 200d); 
 } 
} 

    --基于注解

/** 
 * @Transactional中的的属性 propagation :事务的传播行为 isolation :事务的隔离级别 readOnly :只读 
 *   rollbackFor :发生哪些异常回滚 noRollbackFor :发生哪些异常不回滚 
 *   rollbackForClassName 根据异常类名回滚 
 */ 
@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false) 
public class AccountServiceImpl implements AccountService { 
 // 注入转账的DAO 
 private AccountDao accountDao; 
 
 /** 
 * @param out 
 *  :转出账号 
 * @param in 
 *  :转入账号 
 * @param money 
 *  :转账金额 
 */ 
 @Override 
 public void transfer(String out, String in, Double money) { 
 accountDao.outMoney(out, money); 
 // int i = 1/0; 
 accountDao.inMoney(in, money); 
 } 
 
 public void setAccountDao(AccountDao accountDao) { 
 this.accountDao = accountDao; 
 } 
} 

applicationContext4.xml

<!-- 引入外部的属性文件 --> 
 <context:property-placeholder location="classpath:jdbc.properties"/> 
 
 <!-- 配置c3p0连接池 --> 
 <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> 
 <property name="driverClass" value="${jdbc.driverClass}" /> 
 <property name="jdbcUrl" value="${jdbc.url}" /> 
 <property name="user" value="${jdbc.username}" /> 
 <property name="password" value="${jdbc.password}" /> 
 </bean> 
 
 <!-- 配置业务层类 --> 
 <bean id="accountService" class="com.zs.spring.demo4.AccountServiceImpl"> 
 <property name="accountDao" ref="accountDao" /> 
 </bean> 
 
 <!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> 
 <bean id="accountDao" class="com.zs.spring.demo4.AccountDaoImpl"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- ==================================4.使用注解配置声明式事务============================================ --> 
 
 <!-- 配置事务管理器 --> 
 <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 
 <property name="dataSource" ref="dataSource" /> 
 </bean> 
 
 <!-- 开启注解事务 --> 
 <tx:annotation-driven transaction-manager="transactionManager"/> 

测试:

@RunWith(SpringJUnit4ClassRunner.class) 
@ContextConfiguration("classpath:applicationContext4.xml") 
public class TransactionTest { 
 /** 
 * 一定要注入代理类:因为代理类进行增强的操作 
 */ 
 @Resource(name = "accountService") 
 private AccountService accountService; 
 
 @Test 
 public void demo1() { 
 accountService.transfer("aaa", "bbb", 200d); 
 } 
} 

具体代码和数据库文件参考项目完整代码:

http://xiazai.jb51.net/201805/yuanma/Spring-transaction_jb51.rar

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

相关文章

  • Mybatis Plus框架项目落地实践分析总结

    Mybatis Plus框架项目落地实践分析总结

    这篇文章主要为大家介绍了Mybatis Plus框架项目落地实践分析总结,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • 关于Guava缓存详解及使用说明

    关于Guava缓存详解及使用说明

    这篇文章主要介绍了关于Guava缓存详解及使用说明,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-10-10
  • SpringMVC获取请求参数的方法详解

    SpringMVC获取请求参数的方法详解

    这篇文章主要为大家详细介绍了SpringMVC中获取请求参数的方法,例如通过ServletAPI获取和通过控制器方法的形参获取请求参数等,需要的可以参考下
    2023-07-07
  • Java调用阿里身份证实现验证接口

    Java调用阿里身份证实现验证接口

    这篇文章主要为大家详细介绍了Java如何调用阿里身份证实现验证接口,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-06-06
  • SpringBoot动态定时任务、动态Bean、动态路由详解

    SpringBoot动态定时任务、动态Bean、动态路由详解

    这篇文章主要介绍了SpringBoot动态定时任务、动态Bean、动态路由详解,之前用过Spring中的定时任务,通过@Scheduled注解就能快速的注册一个定时任务,但有的时候,我们业务上需要动态创建,或者根据配置文件、数据库里的配置去创建定时任务,需要的朋友可以参考下
    2023-10-10
  • Springboot事件监听与@Async注解详解

    Springboot事件监听与@Async注解详解

    这篇文章主要介绍了Springboot事件监听与@Async注解详解,在开发中经常可以利用Spring事件监听来实现观察者模式,进行一些非事务性的操作,如记录日志之类的,需要的朋友可以参考下
    2024-01-01
  • Java如何根据不同系统动态获取换行符和盘分割符

    Java如何根据不同系统动态获取换行符和盘分割符

    这篇文章主要介绍了Java如何根据不同系统动态获取换行符和盘分割符,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • SpringBoot 2.x 整合Lombok的方法示例

    SpringBoot 2.x 整合Lombok的方法示例

    Spring Boot是非常高效的开发框架,lombok是一套代码模板解决方案,将极大提升开发的效率,这篇文章主要介绍了SpringBoot 2.x 整合Lombok的方法示例,感兴趣的小伙伴们可以参考一下
    2018-06-06
  • Java中数字黑洞实现代码

    Java中数字黑洞实现代码

    这篇文章主要介绍了Java编程中如何实现数字黑洞算法游戏,其中涉及到了数组、scanner、if语句等Java编程的基础知识,需要的朋友可以参考下
    2017-09-09
  • Java实现map转换成json的方法详解

    Java实现map转换成json的方法详解

    这篇文章主要为大家详细介绍了Java语言实现map转换成json的几种方法,文中的示例代码讲解详细,对我们学习Java有一定帮助,需要的可以参考一下
    2022-05-05

最新评论