way-to-architect
  • 前言
  • Java
    • Java关键字
      • Java中四种修饰符的限制范围
      • static和final
    • 容器
      • 容器概述
        • 容器:综述
        • Iterator原理及实现
        • fast-fail机制
        • 比较器Comparator
        • Collections工具类
      • List
        • List综述
        • ArrayList原理分析
        • ArrayList在循环过程中删除元素的问题
        • 常用的小技巧
        • CopyOnWrite
      • Set
        • Set综述
        • HashSet
        • LinkedHashSet
        • TreeSet
      • Queue
        • Queue综述
        • ArrayBlockingQueue实现原理
        • LinkedBlockingQueue实现原理
        • 高性能无锁队列Disruptor
      • Map
        • Map综述
        • HashMap
          • HashMap实现原理
          • HashMap中的位运算
          • HashMap其他问题
        • LinkedHashMap
        • TreeMap
        • ConcurrentHashMap
          • ConcurrentHashMap实现原理JDK1.7
          • ConcurrentHashMap实现原理JDK1.8
        • ConcurrentSkipListMap
        • Map中key和value的null的问题
    • 线程
      • 线程
        • 创建线程
        • 线程状态及切换
        • 线程中断的理解
        • 几种方法的解释
        • 用户线程与守护线程
        • 线程组ThreadGroup
      • 线程池
        • 线程池工作原理及创建
        • Executor
        • 如何确保同一属性的任务被同一线程执行
      • ThreadLocal
        • ThreadLocal原理
        • ThreadLocal之父子线程传值
        • InheritableThreadLocal
      • 同步与锁
        • 线程安全与锁优化
        • synchronize关键字
        • Lock
          • 队列同步器
            • 同步状态的获取与释放
            • 使用方式
            • 示例:Mutex
            • 示例:TwinsLock
          • 重入锁和读写锁
          • LockSupport
          • Condition
          • 并发工具类
        • CAS
          • CAS的理解
          • Java中原子操作类
        • 3个经典同步问题
      • fork/join的理解
    • I/O
      • I/O概述
        • 磁盘I/O与网络I/O
        • 主要接口
        • 输入流和输出流的使用示例
        • InputStream的重复读
        • BufferdxxxxStream
        • Serailizable
        • File常用方法
        • Files和Path
        • RandomAccessFile
        • 通过零拷贝实现有效数据传输
        • 正确地处理文件
      • NIO基础
      • NIO2
      • Netty
        • Java I/O的演进之路
        • 为什么是Netty
        • 更多
      • I/O调优
    • 异常
      • 异常体系及为什么要有这种异常设计
      • 多catch的执行情况
      • try catch finally 与reture
      • 异常处理的误区
      • Preconditions:方法入参校验工具
    • 枚举
      • 常见用法
      • 枚举类在序列化中的问题
    • 注解
      • 概述
      • Spring中的组合注解的条件注解
      • 常用注解
        • JSR-330标准注解
    • 反射
      • 概述
      • 内部类的反射
      • 反射中需要注意的地方
    • 流程控制
      • switch case without break
      • Java: for(;;) vs. while(true)
    • JVM
      • JVM内存结构
      • Java内存模型
      • 垃圾收集器和内存分配策略
      • 四种引用类型区别及何时回收
      • 类文件结构
      • 类初始化顺序
      • 类加载机制
      • 虚拟机执行引擎
      • 逃逸分析
      • JVM常用配置
      • GC日志分析
      • Java8 JVM 参数解读
      • 垃圾收集器和内存分配策略
    • 面向对象
      • Object类中的方法
      • Class类中的方法
      • 值传递还是引用传递?
      • 接口和抽象类的区别
      • 深拷贝和浅拷贝
      • Integer.parseInt()与Interger.valueof()
      • hashCode()与equal()
      • String
        • String池化及intern()方法的作用
        • 关于字符串
    • 序列化
      • Java序列化的方式有哪些?
    • 新特性
      • 流 Stream
        • Stream是什么
        • Stream API详解
        • Stream进阶
        • 流编程
        • 其他事项
      • lambda表达式
      • 默认方法(Default Methods)
      • @FunctionalInterface注解
    • SPI
      • 理解SPI
    • 字节码
      • javaagent
      • 字节码操纵
      • 如何查看类编译后的字节码指令
      • 字节码指令有哪些
  • Python
    • 异常处理
  • Go
  • 数据结构与算法
    • 数据结构
      • 概述
        • 线性表
        • 栈
        • 队列
        • 串
        • 树
        • 图
      • Java的一些实现
      • 红黑树
      • 双缓冲队列
      • 跳表SkipList
    • 算法
      • 概述
      • 常见算法
        • 基本排序
        • 高级排序
        • 动态规划
  • 框架或工具
    • Spring
      • Spring基础
        • Spring整体架构
        • 什么是IoC
        • Ioc容器的基本实现
        • Spring的MainClass
          • Spring的BeanFactory
          • Spring的Register
          • Spring的Resource和ResourceLoader
          • Spring的PropertySource
          • Spring的PropertyResolver
          • Spring的PropertyEditor
          • Spring的Convert
          • Spring的BeanDefinition
          • Spring的BeanDefinitionReader
          • Spring的BeanDefiniton其他Reader
          • Spring的BeanDefinition其他Reader2
          • Spring的Aware
          • Spring的BeanFctoryPostProcessor
          • Spring的BeanPostProcessor
          • Spring的Listener
        • Xml格式的应用启动
          • Xml格式的应用启动2
          • Xml格式的应用启动3
          • Xml格式的应用启动4
          • Xml格式的应用启动5
          • Xml格式的应用启动6
          • Xml格式的应用启动7
        • Spring中的设计模式
        • 什么是AOP
        • Spring中AOP的实现
      • Spring应用
        • Spring的事务控制
        • @Transactional注解在什么情况下会失效
        • 如何在数据库事务提交成功后进行异步操作
        • Spring中定时任务的原理
    • SpringMVC
      • Controller是如何将参数和前端传来的数据一一对应的
      • 请求处理流程
    • Zookeeper
      • Zookeeper是什么
      • Zookeeper能干啥
    • Shiro
    • druid
    • Netty
    • Consul
      • Consul是什么
    • etcd
    • confd
    • Akka
      • Actor模型是什么
  • 数据库
    • 基本概念
    • MySQL
      • 基本配置
      • MySQL数据类型
      • MySQL存储引擎
      • MySQL事务
        • MySQL事务概念
      • MySQL索引
        • MySQL中的索引类型
        • B-Tree/B+Tree概述
        • 为什么使用B+Tree
        • MySQL中的B+Tree索引
        • MySQL高性能索引策略
      • MySQL查询
        • MySQL查询过程
        • MySQL查询性能优化
        • 使用EXPLAIN
      • MySQL锁
        • MySQL中锁概述
        • InnoDB的并发控制
        • MySQL乐观锁
      • MySQL分库分表
        • 分库/分表
        • 跨库JOIN
        • 跨库分页
        • 分库分表后的平滑扩容
        • 分区表
        • 分布式ID生成方法
      • MySQL实战
        • 在线表结构变更
        • MySQL优化规则
        • MySQL问题排查
        • 常见查询场景
    • Redis
    • Hbase
    • OpenTSDB
    • rrd
    • MongoDB
    • 连接池
  • 系统设计
    • 一致性Hash算法
    • 限流
      • 限流是什么
      • 限流算法
      • 应用内限流
      • 分布式限流
      • 接入层限流
        • ngx_http_limit_conn_module
        • ngx_http_limit_req_module
        • lua_resty_limit-tarffic
      • 节流
    • 降级
      • 降级详解
      • 人工降级开关的实现
      • 自动降级的实现:Hystrix
    • 负载均衡
      • 概述
      • 互联网架构下的负载均衡
      • Nginx负载均衡(七层)
      • Nginx负载均衡(四层)
      • Nginx动态配置
    • 超时与重试机制
      • 什么地方要超时与重试
      • 代理层超时与重试
      • Web容器超时
      • 中间件客户端超时与重试
      • 数据库超时
      • NoSQL客户端超时设置
      • 业务超时
      • 前端请求超时
    • 网关
    • CAP
      • 什么是CAP
      • CAP理解
    • 生产者-消费者模型
      • 使用notify/wait方式
      • 使用await/signal实现
      • 使用阻塞队列实现
      • 使用信号量实现
      • 使用管道流实现
      • 无锁队列Disruptor
      • 双缓冲队列
    • 缓存
      • 缓存概述
      • 数据库缓存
      • 应用缓存
      • 前端缓存
      • 本地缓存
    • 秒杀
    • LRU
  • 版本控制
    • Git
      • Git常用命令
      • 场景命令
    • Svn
  • 计算机操作系统
    • Linux
      • Linux中重要概念
      • 常用命令
      • 查看日志
      • 权限管理
      • 登录或传输
      • 防火墙
      • 配置ssh免密
      • 进程
      • 防火墙
    • Mac
    • 计算机基础
      • 进制
      • Java中的位运算
      • 计算机存储系统结构
  • 网络
    • TCP三次握手和四次挥手
    • 网络术语
      • 网关、路由器、交换机、IP等
      • VLAN
      • LAN
  • 设计模式
    • 设计模式概述
    • 创建型
      • 单例模式
      • 工厂模式
      • 建造者模式
      • 原型模式
      • 享元模式
    • 行为型
      • 观察者模式
      • 策略模式
      • 模板模式
      • 责任链模式
      • 命令模式
      • 外观模式
      • 迭代器模式
      • 中介者模式
        • 中介模式续
      • 状态模式
        • 状态模式实例
        • 状态模式思考
      • 访问者模式
        • 访问者实例1
        • 访问者模式续
    • 结构型
      • 组合模式
        • 组合模式续
      • 装饰模式
        • 装饰模式续
      • 代理模式
      • 备忘录模式
      • 桥接模式
        • 桥接模式实例一
  • 构建工具
    • Maven
      • 常用命令
      • Maven生命周期
      • Maven中的变量和属性
      • 不同环境的如何配置不同的变量
      • 常用插件及配置
      • 其他问题
      • dependencies与dependencyManagement的区别
    • Gradle
  • 大数据
    • Hadoop
    • Storm
    • Spark
  • 服务器
    • Tomcat
      • server.xml配置详解
      • 线程池和连接数配置
      • Maven远程部署
      • 一些小技巧
      • Tomcat类加载机制分析
      • Tomcat的日志
      • Tomcat架构
        • 概述
        • Server 的启动流程
        • 请求处理流程
    • Nginx
      • 常用命令
      • 基本配置
      • Lua
    • Tengine
  • 中间件
    • 任务调度
      • 为什么需要任务调度
    • 消息队列
      • 为什么需要消息队列
      • 消息队列关键点
      • 消息中间件需要解决的问题
      • 不同消息队列产品对比
      • RocketMQ
        • 快速入门
        • 整体架构
        • 部署方式
          • Broker部署方案
        • 客户端使用
          • 客户端使用指南
          • 快速开始
          • 简单示例
          • 有序消息示例
          • 广播消息示例
          • 定时消息示例
          • 批量消息示例
          • 过滤消息示例
          • 日志输出配置示例
        • 关键点实现
          • 顺序消息的实现
        • 最佳实践
          • Broker的最佳实践
          • 生产者最佳实践
            • 生产者最佳实践续
          • 消费者最佳实践
            • 消费者最佳实践续
          • 名称服务最佳实践
          • JVM/kernel配置的最佳实践
          • 新特性 Filter Server
          • 其他事项
      • RabbitMQ
      • Kafka
    • 分布式事务
      • 什么是分布式事务
      • 解决方案
    • 服务治理
      • RPC概念
      • RPC最简实现
      • 为什么需要服务治理
      • Dubbo
        • Dubbo整体架构
      • Java RMI
    • 分布式锁
      • 如何设计分布式锁
        • 基于zookeeper
        • 基于Redis
    • 注册中心
      • 注册中心的职责
      • 不同注册中心的比较
    • 配置中心
      • 概述
      • 配置中心的实现与选型
  • Web开发
    • Http请求类型及区别
    • 常见的content-type
    • 如何处理跨域
    • Restful最佳实践
    • HTTP状态码
    • Http下载原理
  • 测试
    • 压测:apache bench
    • 压测:Jmeter
Powered by GitBook
On this page
  • 注解位置
  • @Transactional的属性
  • 是否只读(readOnly)
  • 传播行为(propagation)
  • 事务隔离级别(isolation)
  • 回滚(rollbackFor和noRollbackFor)
  • 参考
  1. 框架或工具
  2. Spring
  3. Spring应用

Spring的事务控制

注解位置

@Transactional可以放在两个位置

Service层的实现类上:该类中所有方法都进行事务处理

@Service
@Transactional
public class OrgServiceImpl implements OrgService {
          ...
}

Service层的实现类的具体方法上:该方法进行事务处理

@Service
public class OrgServiceImpl implements OrgService {
    ...
    @Override
    @Transactional
    public int addOrg(Org org) {
        ...
    }
    ...
}

@Transactional的属性

先看一个表格总结:

属性名

功 能 描 述

readOnly

设置当前事务是否为只读事务,默认false

propagation

设置事务的传播行为,默认为Propagation.REQUIRED

isolation

设置底层数据库的事务隔离级别。默认值为Isolation.DEFAULT

timeout

设置事务的超时秒数,默认值为-1,永不超时

rollbackFor

设置需要进行回滚的异常类数组,当方法中抛出指定异常数组中的异常时,则进行事务回滚。

rollbackForClassName

设置需要进行回滚的异常类名称数组,当方法中抛出指定异常名称数组中的异常时,则进行事务回滚。

noRollbackFor

设置不需要进行回滚的异常类数组,当方法中抛出指定异常数组中的异常时,不进行事务回滚。

noRollbackForClassName

设置不需要进行回滚的异常类名称数组,当方法中抛出指定异常名称数组中的异常时,不进行事务回滚。

再来看几个重要属性的具体释义。

是否只读(readOnly)

是否是只读事务。

取值

含义

false(默认)

说明为读写事务

true

说明为只读事务,对于JDBC而言,只读事务会有一定的速度优化。此时,事务控制的其他配置会采用默认值,事务的隔离级别(isolation) 为DEFAULT(采用底层数据源的隔离级别),事务的传播行为(propagation)则是REQUIRED,所以还是会有事务存在。

1 readonly并不是所有数据库都支持的,不同的数据库下会有不同的结果。 2 设置了readonly后,connection都会被赋予readonly,效果取决于数据库的实现。 3 在ORM中,设置了readonly会赋予一些额外的优化,例如在Hibernate中,会被禁止flush等。

在将事务设置成只读后,相当于将数据库设置成只读数据库,此时若要进行写的操作,会出现错误。如果你使用的是mysql数据库,出现了这样的报错信息,那么很有可能是你在增删改等修改操作的方法上不小心加上了readOnly=true。

Caused by: java.sql.SQLException: Connection is read-only. Queries leading to data modification are not allowed
     at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:910)
     at com.mysql.jdbc.PreparedStatement.execute(PreparedStatement.java:792)

在增删改方法中,采用默认readOnly=false即可,也就是不需要写这个属性。 在查询方法中,对于是否配置readOnly=true,目前我是配置了的,不过还有待研究。。。

传播行为(propagation)

事务的传播行为,有以下几种取值。

取值

含义

Propagation.REQUIRED(默认)

如果有事务则加入事务,没有的话就新建一个事务

Propagation.REQUIRES_NEW

不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务

Propagation.MANDATORY

必须在一个已有的事务中执行,否则抛出异常

Propagation.NEVER

必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反)

Propagation.SUPPORTS

如果其他bean调用这个方法,在其他bean中声明事务,那就用事务;如果其他bean没有声明事务,那就不用事务

Propagation.NOT_SUPPORTED

不为这个方法开启事务

Propagation.REQUIRED

ServiceA {   
     @Transactional(propagation = Propagation.REQUIRED)
     void methodA() {   
         ServiceB.methodB();   
     }   
}   

ServiceB {   
     @Transactional(propagation = Propagation.REQUIRED)
     void methodB() {  

     }   
}  

public class Controller(){
     public void methodC(){
          ServiceA.methodA();   
     }
}

这种情况下:methodA会开启新的事务,methodB会加入这个事务之中。无论在何时何处发生异常,两者都会回滚。

业务应用中绝大部分都是这种场景,所以Spring将其设为了默认值。

PROPAGATION_REQUIRES_NEW

ServiceA {   
     @Transactional(propagation = Propagation.REQUIRED)
     void methodA() {   
         ServiceB.methodB();   
     }   
}   

ServiceB {   
     @Transactional(propagation = Propagation.PROPAGATION_REQUIRES_NEW)
     void methodB() {  

     }   
}  

public class Controller(){
     public void methodC(){
          ServiceA.methodA();   
     }
}

这种情况下:methodA会开启新的事务,当执行到methodB的时候,methodA所在的事务就会被挂起,methodB会开启一个新的事务,等待methodB的事务提交以后,methodA才继续执行。

回滚情况:

  • methodB已经提交,methodA发生异常,则回滚methodA,而methodB不回滚

  • methodB发生异常(未提交),则methodB都回滚

    • 如果methodA内部catch了methodB的异常,methodA没有发生异常,methodA可以成功提交,不会回滚

    • 如果methodA内部没有catch住methodB的异常,methodA会回滚

PROPAGATION_SUPPORTS

如果当前环境有事务,就加入到当前事务;如果没有事务,就以非事务的方式执行。听起来跟普通方法没什么两样,它与普通方法的区别如下:

  • 加了PROPAGATION_SUPPORTS的方法可以获取和当前事务环境一致的Connection或Session,而普通方法获取到的是最新的;

  • 加了PROPAGATION_SUPPORTS的方法可以在挂起事务、恢复事务的时侯执行回调方法,而普通方法做不到。

Spring的文档是这么说的

NOTE:For transaction managers with transaction synchronization,PROPAGATION_SUPPORTSis slightly different from no transaction at all, as it defines a transaction scope that synchronization might apply to. As a consequence, the same resources (a JDBCConnection, a HibernateSession, etc) will be shared for the entire specified scope. Note that the exact behavior depends on the actual synchronization configuration of the transaction manager!

In general, usePROPAGATION_SUPPORTSwith care! In particular, do not rely onPROPAGATION_REQUIREDorPROPAGATION_REQUIRES_NEWwithinaPROPAGATION_SUPPORTSscope (which may lead to synchronization conflicts at runtime). If such nesting is unavoidable, make sure to configure your transaction manager appropriately (typically switching to "synchronization on actual transaction").

事务隔离级别(isolation)

事务隔离级别,有以下几种取值:

取值

含义

Isolation.DEFAULT(默认)

使用底层数据源的配置(下面4种之一)

Isolation.READ_UNCOMMITTED

读取未提交数据(会出现脏读, 不可重复读) 基本不使用

Isolation.READ_COMMITTED

读取已提交数据(会出现不可重复读和幻读)

Isolation.REPEATABLE_READ

可重复读(会出现幻读)

Isolation.SERIALIZABLE

串行化

除非有特殊需求一般不需要设置显式这个属性,即采用默认值,让Spring使用数据库设置的事务隔离级别。

回滚(rollbackFor和noRollbackFor)

默认是,当抛出一个unchecked异常(也就是运行时异常RuntimeException或其子类例的实例)时,会进行事务回滚。从事务方法中抛出的Checked exceptions将不被标识进行事务回滚。

概念补充:什么是unchecked和checked异常

unchecked:运行时异常RuntimeException或其子类例的实例都是unchecked异常,可以理解为那些不用写try-catch的异常。 checked:除了上面说的,都是checked异常

如何改变默认规则: a.让checked异常也回滚:设置@Transactional(rollbackFor=Exception.class) b.让unchecked异常不回滚: 设置@Transactional(notRollbackFor=RunTimeException.class)

参考

PreviousSpring应用Next@Transactional注解在什么情况下会失效

Last updated 6 years ago

中给出的结论是:

Spring 事务 readOnly 到底是怎么回事
Spring 事务 readOnly 到底是怎么回事?
MySQL学习笔记(二):事务管理
使用@Transactional(propagation = Propagation.SUPPORTS)和不加@Transactional 有什么区别?