使用Spring管理数据库事务
在整个JavaWeb项目开发中,事务是用来开发可靠性网络应用程序的最关键部分。当应用程序与后端资源进行交互时,就会用到事务,这里的后端资源包括数据库、MQ、ERP等。而数据库事务是最常见的类型,而我们常说的事务也就是狭义上的与关系型数据库交互的事务。
事务主要分为本地事务和全局事务。全局事务又称分布式事务,本地事务就是当应用程序连接单个数据库资源时的事务,也是本文化主要讨论的内容。
一、事务的一些基本概念
事务的属性(ACID):
- 原子性
 - 一致性
 - 隔离性
 - 持久性
 
白话“事务”
事务有三个状态(或者说是过程):开始、提交、回滚。
假设有这么一个场景:张三和李四各有100元,有一天,张三要给李四转10元。
相当于目前的微信转账,张三给李四发了10元的转账。有以下三种状态

上边这个例子有一处不恰当的地方就是,就算李四没有操作这10元时,张三已经少了10元,这一点和事务有出入 ,我们就假装如果李四不接收或者退回这10元,张三的微信钱包里还有100元。但是在微信中有那么多的人相互转账,每一次转账就是一个事务,我们就要把这些事务进行隔离,但是它有不同的隔离级别(见下)
事务的隔离级别
| 隔离级别 | 描述 | 举例 | 
|---|---|---|
| DEFAULT | 底层数据库存储的默认隔离级别 | |
| READ_UNCOMMITTED | 最低的隔离级别,可以说它并不是事务,因为它允许其他事务来读取未来提交的数据 | 上边的例子中,就算李四没有收这10元,其他人也能读取到李四多了10元。 | 
| READ_COMMITTED | 大多数数据库的默认级别,它确保其他事务可以读取其他事务已经提交的数据 | 只有当李四对这10元进行操作(接收或者退回)时,别人才能看到这两个的余额变化。 | 
| REPEATABLE_READ | 比上一个更为严格,它确保在选择了数据后,如果其他事务对这个数据进行了更改,就可以选择新的数据。 | 上边的是在转账过程中,就算别人给张三又转了10元,在这个事务提交前,张三一直认为自己只有100元。但是这个类型中,张三在转账过程中,可以查到自己有110元 | 
| SERIALIZABLE | 可序列化,是最严格最可靠的隔离级别,让所有事务一个接着一个地运行 | 系统让每个人转账事务一个一个地执行,就不会有任何错误了(当然,这里的事务不单单指转账这一个事务) | 
事务的传播类型
也就是当前事务开始的机制和时间,相当于这么多的人之间的微信转账应该怎么进行
| 传播类型 | 描述 | 
|---|---|
| PROPAGATION_REQUIRED | 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中; | 
| PROPAGATION_SUPPORTS | 支持当前事务,如果当前没有事务,就以非事务方式执行; | 
| PROPAGATION_MANDATORY | 使用当前的事务,如果当前没有事务,就抛出异常; | 
| PROPAGATION_REQUIRES_NEW | 新建事务,如果当前存在事务,把当前事务挂起; | 
| PROPAGATION_NOT_SUPPORTED | 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起; | 
| PROPAGATION_NEVER | 以非事务方式执行,如果当前存在事务,则抛出异常; | 
| PEOPAGATION_NESTED | 如果当前存在事务,则在潜逃事务内执行。如果当前没有事务,则执行PROPAGATION_REQUIRED列斯的操作; | 
二、Spring中解决事务问题
在Spring中解决事务问题有两种:声明式事务和编程式事务(不建议使用)
Spring中支持事务的最底层接口是PlatformTransactionManager,而我们使用的只能它的子类
public interface PlatformTransactionManager {
    //获取事务状态
    TransactionStatus getTransaction(@Nullable TransactionDefinition var1) throws TransactionException;
	//提交
    void commit(TransactionStatus var1) throws TransactionException;
	//回滚
    void rollback(TransactionStatus var1) throws TransactionException;
}
这个接口中主要用了TransactionDefinition和TransactionStatus两个类。有兴趣的可以看一下。下边这是它的子类图,我们这里使用的是DataSourceTransactionManager作为事务管理类,不管使用何种方式,PlatformTransactionManager这个接口的子类一定要有。

1. 声明式事务
使用注解
配置文件如下:
<!--引入公共的配置文件-->
<import resource="application-context.xml"/>
<!--Spring提供的事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <constructor-arg ref="dataSource"/>
</bean>
<!--
    开启事务注解
    这里有个小技巧,如果你的事务管理bean名不是transactionManager
    就要给这个标签配置transaction-manager来指定
    -->
<tx:annotation-driven/>
<!--    配置spring扫描注解注入service类-->
<context:component-scan base-package="cn.lyn4ever.service"/>
 然后在类或方法上加上这个@Transactional注解就可以
@Transactional
public void insertOne(){
    Store store  =new Store();
    store.setTitle("华为P30");
    storeMapper.insertOne(store);
    int j = 10/0;//指定报错,让事务回滚
}
- 使用AOP配置
 
使用aop的话,我们只需要进行配置,可以对我们写的业务代码无任何侵入。如果对AOP知识不是很了解,可以参考我之前的AOP系列教程Spring学习笔记,AOP的配置也有多种,这里就直接使用aop名称空间了
<import resource="application-context.xml"/>
<!--Spring提供的事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <constructor-arg ref="dataSource"/>
</bean>
<tx:advice id="txAdvice">
    <tx:attributes>
        <!-- 对单独方法配置属性-->
        <tx:method name="insert*" rollback-for="java.lang.Exception"/>
        <tx:method name="*"/>
    </tx:attributes>
</tx:advice>
<aop:config>
    <aop:pointcut id="serviceTrans" expression="execution(* cn.lyn4ever.serviceaop.*.*(..))"/>
    <aop:advisor advice-ref="txAdvice" pointcut-ref="serviceTrans"/>
</aop:config>
<!--    配置spring扫描注解注入service类-->
<context:component-scan base-package="cn.lyn4ever.serviceaop"/>
- 以上两种方式的对比:
- 使用注解能更加细粒度地进行控制,因为并不是所有service方法都需要事务。而使用AOP使用的面向切面编程,所以可以大批量的进行控制,而一般都是在service层进入切入的。
 - 使用注解的话,配置简单,AOP的配置稍微复杂点儿。
 - 如果是新项目的话,建议从一开始就使用注解式开发。如果是更改之前没有用过事务(一般成熟的程序员不会这么干)的项目,或者无法修改源代码的情况下,建设使用AOP。
 - 个人经验,建议使用注解开发,能灵活的配置每一个方法及类。使用AOP的话,有时候调试起来不太方便,如果你的调试内容跨越了一个service方法,会进入aop通知方法,很麻烦。
 
 
2. 编程式事务
顾名思义,就是将事务的操作直接写在业务代码中,这样做最简单,但是最不建议。有两种方式,一种就是将PlatformTransactionManager的实例注入到bean中,使用它。另一种就是使用Spring为我们提供的TransactionTemplate。这里直接使用第二种,这时,我们只需要使用Spring注入注入transactionManager和这两个类,但是为了不和之前的配置混淆,我直接new这两个对象了,也就是说,使用编程式事务,只需要这两个对象就够了,不需要其他任何有关事务的配置,只需要一个数据源
@Autowired
private DataSource dataSource;
@Test
public void fun() {
    DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
    //设置数据源,这个数据源的bean是由Spring提供的
    transactionManager.setDataSource(dataSource);
    TransactionTemplate transactionTemplate = new TransactionTemplate(transactionManager);
    transactionTemplate.execute(txStatus -> {
        Store store = new Store();
        store.setTitle("小米11");
        storeMapper.insertOne(store);
        //制造错误,让事务回滚
        int i = 10 / 0;
        return null;
    });
}
关注微信公众号“小鱼与Java”,回复Spring获取代码地址和更多学习资料

使用Spring管理数据库事务的更多相关文章
- Spring管理Hibernate事务
		
在没有加入Spring来管理Hibernate事务之前,Hibernate对事务的管理的顺序是: 开始事务 提交事务 关闭事务 这样做的原因是Hibernate对事务默认是手动提交,如果不想手动提交, ...
 - spring管理的事务
		
之前对spring的事务传播机制没有概念,花点时间去看了事务的源码,以及这些事务传播机制使用的文档,在此做一下简单的笔记 正文 下面说提到的共享事务的意思就是几个service共用同一个事务,如传播机 ...
 - Spring管理 hibernate 事务配置的五种方式
		
Spring配置文件中关于事务配置总是由三个组成部分,DataSource.TransactionManager和代理机制这三部分,无论是那种配置方法,一般变化的只是代理机制这块! 首先我创建了两个类 ...
 - spring 管理 jdbc 事务
		
@Transactional 业务实现类 类名上方--这个类中的方法,执行操作前会打开事务. 默认:RuntimeException 自动回滚, 可以try catch 的异常,不会滚 方法名 ...
 - 数据库事务和spring事务的区别
		
数据库事务和spring事务 本质上其实是同一个概念,spring的事务是对数据库的事务的封装,最后本质的实现还是在数据库,假如数据库不支持事务的话,spring的事务是没有作用的.数据库的事务说简单 ...
 - mybatis源码分析(8)-----事务(mybatis管理、spring管理)
		
写在前面 接口:MyBatis的事务Transaction的接口有一下实现类 JdbcTransaction 由jdbc管理的事务(即利用Connection对象完成对事务的提交(commit()). ...
 - spring 声明式事务原理解读
		
在Spring中,声明式事务是通过事务属性(transaction attribute)来定义的.事务属性描述了事务策略如何应用到方法上.事务属性包含5个方面: 传播行为 隔离级别 是否只 ...
 - 8.spring:事务管理(上):Spring的数据库编程、编程式事务管理
		
Spring的数据库编程 Spring框架提供了JDBC模板模式------>JdbcTemplate 简化了开发,在开发中并不经常是使用 实际开发更多使用的是Hibernate和MyBatis ...
 - spring对数据库的操作、spring中事务管理的介绍与操作
		
jdbcTemplate的入门 创建maven工程 此处省略 导入依赖 <!-- https://mvnrepository.com/artifact/org.springframework/s ...
 
随机推荐
- audioContext.decodeAudioData 返回null 错误
			
此问题并不是100%出现.没想到国外大神已经有处理此问题的经验 原贴地址: https://stackoverflow.com/questions/10365335/decodeaudiodata-r ...
 - vue实现选中效果
			
前情提要 好久没有写Vue了,略有生疏,这个东西还是得多用.下午看到一个需求,选择相册图片作为轮播图显示.接口返回相册列表,用户选一下再扔回去.直到我看到e.target.className我就知道这 ...
 - AI广度优先搜索算法,项目实战北京地图/贪心学院
			
广度优先搜索算法详解地铁路线 北京很大,附上地铁图,不要迷路!!! 作为一个程序员,在北京,你很有可能住在回龙观地区,经常从龙泽上地铁,然后畅游北京. 当有一天,你老家的朋友来北京了,希望你能够带她去 ...
 - 再说scss
			
1. CSS预处理器 定义了一种新的专门的编程语言,编译后成正常的CSS文件.为CSS增加一些编程的特性,无需考虑浏览器的兼容问题,让CSS更加简洁,适应性更强,可读性更佳,更易于代码的维护等诸多好处 ...
 - wpf 菜单样式和绑定树形数据
			
前言 在wpf开发中,经常会使用到Menu和ContentMenu.但是原生的样式比较简陋,对于比较追求界面美好的人来说是十分不友好的.那么,这就涉及到对Menu的样式修改了.与此同时,我们还希望Me ...
 - (转)协议森林07 傀儡 (UDP协议)
			
协议森林07 傀儡 (UDP协议) 作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 我们已经讲解了物理层.连接层和网络层.最开始的 ...
 - AVR单片机教程——走向高层
			
本文隶属于AVR单片机教程系列. 在系列教程的最后一篇中,我将向你推荐3个可以深造的方向:RTOS.C++.事件驱动.掌握这些技术可以帮助你更快.更好地开发更大的项目. 本文涉及到许多概念性的内容 ...
 - 可运行jar包的几种打包/部署方式(转)
			
转自:https://www.cnblogs.com/yjmyzz/p/executable-jar.html java项目开发中,最终生成的jar,大概可分为二类,一类是一些通用的工具类(不包含ma ...
 - nuxt创建项目的步骤
			
nuxt创建项目的步骤 1.基本步骤 // 创建package.json依赖管理文件 npm init -y // 在package.json文件中添加运行nuxt的命令,之后npm run dev启 ...
 - Python-时间戳、元组时间的格式、自定义时间格式之间的转换
			
一.时间戳.元组时间的格式.自定义时间格式之间的转换 1.下面是三者之间的转换关系: 2.代码如下: import time import datetime print(time.time()) #获 ...