事务的传播级别

使用Demo:

测试:
controller层:

 @ResponseBody@GetMapping("/save")@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.REPEATABLE_READ)public void save() {System.out.println(employeeService.saveEmployee());System.out.println(employeeService.saveEmployee2());}

servoce层:

@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){Employee employee = new Employee();employee.setName("1");employee.setAge("1");int save = employeeDao.save(employee);System.out.println(employee.getId());return save;}@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.REPEATABLE_READ)public int saveEmployee2(){Employee employee = new Employee();employee.setName("2");employee.setAge("2");int save = employeeDao.save(employee);System.out.println(employee.getId());TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();//手动抛出异常return save;}

REQUIRED传播级别,两条记录都插入失败,如果servoce层两个注解注释掉,也是插入失败。

    @Transactional(propagation = Propagation.SUPPORTS, isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){

SUPPORTS传播级别,两条记录都插入失败,如果controller层去掉注解,第一条记录插入成功(因为以非事务方式运行)。第二条记录插入失败。

    @Transactional(propagation = Propagation.MANDATORY, isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){

MANDATORY传播级别,两条记录都插入失败,如果controller层去掉注解,会抛出异常No existing transaction found for transaction marked with propagation ‘mandatory’ (因为saveEmployee()方法不存在事务)

@Transactional(propagation = Propagation.REQUIRES_NEW, isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){

REQUIRES_NEW传播级别,第一条记录插入成功(因为新建了一个事务,虽然存在REQUIRED事务,但是会将当前事务挂起),第二条记录插入失败。

@Transactional(propagation = Propagation.NOT_SUPPORTED, isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){

NOT_SUPPORTED传播级别,第一条记录插入成功(因为非事务方式运行,虽然存在REQUIRED事务,但是会将当前事务挂起),第二条记录插入失败。

@Transactional(propagation = Propagation.NEVER  , isolation = Isolation.REPEATABLE_READ)public int saveEmployee(){

NEVER 传播级别
两条记录都插入失败。(因为存在controller事务)抛出异常Existing transaction found for transaction marked with propagation ‘never’。
如果把controller中transaction注释掉,第一条记录插入成功(因为没事务),第二条记录插入失败。

我们知道 Spring 事务的原理是 AOP,进行了切面增强,那么失效的根本原因是这个 AOP 不起作用了

1. 事务失效的几种原因

1.1 没有被 Spring 管理

1.2 数据源没有配置事务管理器

1.3 异常被吃了

@Service
public class UserService{@Transactionalpublic void updateUser(User user) {try {System.out.println("--------");//do something} catch {//do something}}
}

1.4 异常类型错误

这个问题在第二问讲过了,因为默认回滚的是:RuntimeException。如果是其他异常想要回滚,需要在 @Transactional注解上加 rollbackFor 属性。

1.5 数据库引擎不支持事务

毕竟 Spring 事务用的是数据库的事务,如果数据库不支持事务,那 Spring 事务肯定是无法生效滴。

1.6 方法不是 public 的

@Transactional 注解的方法都是被外部其他类调用才有效,那么如果方法修饰符是 private 的,这个方法能被外部其他类调到么?
既然调不到,事务生效有意义吗?想通这套逻辑就行了。 记住 :@Transactional 注解只能应用到 public 方法上。如果你在protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错,但是这个被注解的方法将不会加入事务之行。

1.7 同一类中,非事务类,调用事务类,事务会失效

@Service
public class UserService{public void update(User user) {updateUser(user);}@Transactionalpublic void updateUser(User user) {System.out.println("--------");//do something}
}

此时是无效的。因此上面的代码等同于:

@Service
public class UserService{public void update(User user) {this.updateUser(user);}@Transactionalpublic void updateUser(User user) {System.out.println("--------");//do something}
}
二种解决方法:
1,非事务类 设置事务类2.事务类进行代理调用。此时,这个 this 对象不是代理类,而是 UserService 对象本身。 解决方法很简单,让那个 this 变成 UserService 的代理类即可。

2. Spring 什么情况下进行事务回滚

首先我们要明白, Spring 事务回滚机制是这样的:当所拦截的方法有指定异常抛出,事务才会自动进行回滚!
因此,如果你默默的吞掉异常,像下面这样:

@Service
public class UserService{@Transactionalpublic void updateUser(User user) {try {System.out.println("------");//do something} catch {//do something}}
}

那切面捕捉不到异常,肯定是不会回滚的。

还有就是,默认配置下,事务只会对 Error 与 RuntimeException 及其子类这些异常做出回滚。一般的 Exception
这些 Checked 异常不会发生回滚。如果一般的 Exception 想回滚,要做出如下配置:

@Transactional(rollbackFor = Exception.class)

但是在实际开发中,我们会遇到这么一种情况:就是并没有异常发生,但是由于事务结果未满足具体业务需求,所以我们需要手动回滚事务。于是乎方法也很简单:
• 自己在代码里抛出一个自定义异常(常用);
• 通过编程用代码回滚(不常用)。

TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();

3. Spring 事务隔离和数据库事务隔离是不是一个概念

OK,是一回事! 我们先明确一点,数据库一般有四种隔离级别,分别为:
• Read Uncommitted:未提交读;
• Read Committed:提交读、不可重复读;
• Repeatable Read:可重复读;
• Serializable:可串行化。
而Spring 只是在此基础上抽象出一种隔离级别 default,表示以数据库默认配置的为主。例如,MySQL 默认的事务隔离级别为
Repeatable Read,而 Oracle 默认隔离级别为Read Committed。

于是乎,有一个经典问题是这么问的:

  • 我数据库的配置隔离级别是Read Commited,而Spring配置的隔离级别是Repeatable
    Read,请问这时隔离级别是以哪一个为准?

答案是以 Spring 配置的为准。JDBC 有一个接口是这样的:

意思就是,如果 Spring 定义的隔离级别和数据库的不一样,则以Spring 定义的为准。另外,如果 Spring 设置的隔离级别数据库不支持,设置的效果取决于数据库。

4. Spring 事务控制放在 Service 层,在 Service 方法中一个方法调用 Service 中的另一个方法,默认开启几个事务

此题考查的是 Spring 的事务传播行为。 我们都知道,默认的传播行为是 PROPAGATION_REQUIRED。
如果外层有事务,则当前事务加入到外层事务,一起提交并一起回滚;如果外层没有事务,新建一个事务执行。也就是说,默认情况下只有一个事务。

5. 怎么保证 Spring 事务内的连接唯一性

这道题很多种问法,例如 Spring 是如何保证事务获取的是同一个 Connection?
OK,开始我们的讲解。其实答案只有一句话,因为那个 Connection 在事务开始时封装在了 ThreadLocal 里,后面事务执行过程中,都是从 ThreadLocal中 取的。肯定能保证唯一,因为都是在一个线程中执行。

Spring事务Transactional详解相关推荐

  1. Spring事务管理详解_基本原理_事务管理方式

    Spring事务管理详解_基本原理_事务管理方式 1. 事务的基本原理 Spring事务的本质其实就是数据库对事务的支持,使用JDBC的事务管理机制,就是利用java.sql.Connection对象 ...

  2. 关于事务管理的理解和Spring事务管理详解

    转载于:http://www.mamicode.com/info-detail-1248286.html 1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱. 比如你去ATM机取1000 ...

  3. Spring 事务使用详解

    前言 什么是事务?根据 维基百科事务 介绍,数据库事务(简称:事务)是数据库管理系统执行过程中的一个逻辑单位,由一个有限的数据库操作序列构成.简单来说,事务就是将一系列操作当成一个不可拆分的执行逻辑单 ...

  4. Spring事务管理详解

    什么是事务 事务是逻辑上的一组操作,要么都执行,要么都不执行. 需要注意的是:事务能否生效数据库引擎是否支持事务是关键.比如常用的 MySQL 数据库默认使用支持事务的 innodb引擎.但是,如果把 ...

  5. Spring事务原理详解

    一.使用 spring事务开启和使用比较简单,需要有数据源和事务管理器,然后在启动门面类上开启事务,在需要使用事务的地方添加注解就可以了,我们简单做一下回顾. 1.配置数据源 spring.datas ...

  6. Spring事务属性详解

    spring,是一个Java开源框架,是为了解决企业应用程序开发复杂性由Rod Johnson创建的.框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 J2EE 应用程序 ...

  7. Spring 事务原理详解

    一.事务的基本原理 Spring事务 的本质其实就是数据库对事务的支持,没有数据库的事务支持,spring是无法提供事务功能的.对于纯JDBC操作数据库,想要用到事务,可以按照以下步骤进行:     ...

  8. Spring5源码分析系列(九)Spring事务原理详解

    终于等到了B站的薪资沟通电话,美滋滋,本节开始进入Spring数据访问篇,讲解spring事务,文章参考自Tom老师视频. 事务基本概念 事务(Transaction)是访问并可能更新数据库中各种数据 ...

  9. 什么是事务的传播_这么漂亮的Spring事务管理详解,你不来看看?

    事务概念回顾 什么是事务? 事务是逻辑上的一组操作,要么都执行,要么都不执行. 事物的特性(ACID): 原子性: 事务是最小的执行单位,不允许分割.事务的原子性确保动作要么全部完成,要么完全不起作用 ...

最新文章

  1. Sorenson Capital:值得投资的 5 种 AI 技术
  2. .9-Vue源码之AST(5)
  3. 学长毕业日记 :本科毕业论文写成博士论文的神操作20170326
  4. MapReduce:处理数据密集型文本处理–局部聚合第二部分
  5. HttpService远程校验
  6. Yeslab现任明教教主ISE课程前七部分免费发布
  7. L1-046 整除光棍 (20 分)567
  8. mac mysql-share_mac下安装mysql
  9. Apache+Tomcat动静分离
  10. 火星人谚语系列之五:答案将由我在下一分钟给出(心想事成)
  11. 【数据库/数据挖掘/内容检索】 2019年-中国计算机学会推荐国际学术会议和期刊目录(五)
  12. matlab 地理加权回归,混合时空地理加权回归及参数地两步估计.PDF
  13. windows xp下无线网卡断线的问题。
  14. 谈一谈企业部署erp系统的三大时间段
  15. c语言 将数组转化成二叉树
  16. C#:Winform 打字测速程序 Typer
  17. 菜鸟编程:python中实现中英文文字或单词计数(wordcount)
  18. hive_hbase一个综合练习题目总共包括以下部分
  19. 给米钱包、买买分收取用户15%-20%“砍头息”,泸州银行为资金方
  20. centos 6.2 安装intel 显卡驱动

热门文章

  1. 2022邮箱如何发送群发邮件?邮件群发平台软件哪个好?一文get群发小技巧
  2. powershell执行c语言文件,Windows Powershell 执行文件和脚本
  3. 实战讲解Spring定时任务:@Scheduled(图+文+源码)
  4. 把Windows PC打造成最基本的路由器
  5. 我们真的误解了人工智能?
  6. 计算机式表白方式,6种表白方法 有几种表白方式
  7. QGraphicsView绘画曲线注意点(二)
  8. 毕业设计-基于微信小程序的软考刷题系统
  9. 微帧科技对于AV1在RTC实时场景中的现状与展望
  10. MQTT协议连接阿里云