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>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Spring的编程式事务和声明式事务详解

    入口(了解一些基本概念) Spring事务属性(事务的属性有哪些?) 我们都知道事务有开始,保存点,提交,回滚,隔离级别等属性.那么Spring对于事务属性定义有哪些呢?通过TransactionDefinition接口我们可以了解到: public interface TransactionDefinition{ int getIsolationLevel(); int getPropagationBehavior(); int getTimeout(); boolean isReadOnly

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

    Spring事务管理 Spring支持两种方式的事务管理: 编程式事务管理: 通过Transaction Template手动管理事务,实际应用中很少使用, 使用XML配置声明式事务: 推荐使用(代码侵入性最小),实际是通过AOP实现 实现声明式事务的四种方式: 基于 TransactionInterceptor 的声明式事务: Spring 声明式事务的基础,通常也不建议使用这种方式,但是与前面一样,了解这种方式对理解 Spring 声明式事务有很大作用. 基于 TransactionProx

  • Spring的连接数据库以及JDBC模板(实例讲解)

    前言 今天介绍的是关于Spring的数据库连接以及Jdbc模板的相关API方法,虽然在学习了hibernate之后,会知道实现数据库连接一般都是使用hibernate等持久化框架来实现的.但是,很多时候一些涉及到事务的东西使用这些框架并不能够实现,所以我们还需要结合spring来实现相关的需要. 一.创建工程.导包 要想使用Spring的jdbc模板前,还需要导入相关的jar包: 二.进行相关的bean的创建以及工具类的编写 2.1在数据库中创建了两张表,使用spring的jdbcTemplat

  • spring boot+mybatis 多数据源切换(实例讲解)

    由于公司业务划分了多个数据库,开发一个项目会同事调用多个库,经过学习我们采用了注解+aop的方式实现的 1.首先定义一个注解类 @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface TargetDataSource { String value();//此处接收的是数据源的名称 } 2.然后建一个配置类,这个在项目启动时会加载数据源,一开始采用了HikariCP,查资料说是最快性能最好的

  • Vue编程式跳转的实例代码详解

    编程式跳转的实现代码,如下所示: <template> <ul class = "prolist"> <!-- //产品 --> <!-- :to = "/detail/item.id" --> <!-- 声明式跳转 :to = "{ name: 'detail',params: { id: item.id } }" --> <!-- <router-link :to = &

  • 对Linux下shell编程之for循环的实例讲解

    linux 下 for 循环中可以使用 break 和 continue 关键字来跳出循环, 和java 用法一致 一.常用for循环结构 #语法一 for 变量 in 值1 值2 值3.. do 程序块儿 done # #语法二 for 变量 `命令` do 程序块儿 done # #语法三 for ((初始值; 循环控制; 变量变化)) do 程序块儿 done 二.常用测试结构 1. 输出3次uptime #!/bin/bash for i in 1 2 3 4 5 do echo "$i

  • Spring AOP实现声明式事务机制源码解析

    目录 一.声明式全局事务 二.源码 三.小结: 一.声明式全局事务 在Seata示例工程中,能看到@GlobalTransactional,如下方法示例: @GlobalTransactional public boolean purchase(long accountId, long stockId, long quantity) { String xid = RootContext.getXID(); LOGGER.info("New Transaction Begins: " +

  • 详解Spring学习之声明式事务管理

    前言 在前面的小节中,我们学习了关于事务的概念以及事务管理的重要性,并且通过编程使用Spring的编程式事务管理进行操作,加深对事务管理的重要性的学习,不过,由于编程式的事务管理使用起来不是很方便,所以在日常的开发中基本不怎么使用,接下来的内容我们将学习使用Spring的声明式事务管理,这里有一个地方需要明白的是,Spring的声明式事务管理的实现方式其实是通过AOP的方式来实现的,也就是为原始的事务管理对象创建代理对象,从而实现事务管理增强的 基于TransactionProxyFactory

  • Spring注解 TX声明式事务实现过程解析

    环境搭建导入 maven依赖 <!--spring提供的数据库操作工具--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>5.0.2.RELEASE</version> </dependency> <!--c3p0 数据库连接池--> &

  • vue-router实现编程式导航的代码实例

    除了使用 <router-link> 创建 a 标签来定义导航链接,我们还可以借助 router 的实例方法,通过编写代码来实现.即:通过js动态的进行导航链接. 一.this.$router.push( ) router.push(location, onComplete?, onAbort?) 注意:在 Vue 实例内部,你可以通过 $router 访问路由实例.因此你可以调用 this.$router.push. 想要导航到不同的 URL,则使用 router.push 方法.这个方法会

随机推荐