事务ACID的理解

引入事务的主要目的:

  • 保证数据库从一个一致性状态切换为另一种一致性状态
  • 所有修改要么都保存,要么都不保存
A 原子性

原子性关注单个事务的整体性,需要保证事务中的全部操作是一个单元,要么都成功,要么都失败。

C 一致性

一致性关注,事务开始和结束后,数据库的完整性约束没有被破坏。简单理解,如果有外键的存在,事务执行前,外键可以保证一致性约束;事务执行之后,这个一致性约束也不应该被破坏。

I 隔离性

隔离性关注多个事务之间是否彼此互相不知晓,不影响。

没有隔离性带来的问题

没有隔离级别的时候,多个事务互相交替执行,会出现一些问题:

  • 脏读:读取到其他事务未提交的数据
  • 不可重复读:在事务的开始和结束这段时间,对同一行数据,读取到的值是不同的
  • 幻读:事务开始和结束这段时间,同一个查询,查询到的数据行数不一致
四种隔离级别

四种隔离级别,可以分别解决不同的隔离性问题:

  • 读未提交:事务可以读到其他事务未提交的数据

    • 有脏读,不可重复读,幻读问题
  • 读已提交:事务可以读到其他事务已经提交的数据
    • 解决脏读问题,仍有不可重复读,幻读问题
  • 可重复读:事务在开始到结束这段时间,读到的同一行数据是一致的。
    • 解决脏读、不可重复读,仍有幻读问题
  • 串行化:事务和事务之间是串行执行的
    • 所有问题都可以解决

这四种隔离级别的隔离读,从上到下越来越高,但是性能从上到下越来越低。

隔离性的实现

实现上,数据库会创建一个视图,访问的时候以这个视图的逻辑结果为准。

  • 读未提交:不创建视图
  • 读已提交:在事务中的每条sql语句执行前创建视图
  • 可重复读:事务开始时创建,整个事务执行过程中都已这个视图为准
  • 串行化:直接使用加锁的方式实现

在每一次更新操作以后,会记录一条回滚日志,当前的最新值,可以通过回滚日志回滚到之前的某一个状态,MVCC(多版本并发控制)就是通过回滚段实现的,同一条记录在系统中存在多个版本,不同时刻启动的事务,查看到的就是不同的版本值,并且版本之间互不干扰。

回滚日志是会被删除的,当系统判断,没有事务会用到这些回滚日志的时候就会删除回滚日志,也就是说当这个系统中没有比这个回滚日志更早的事务视图的时候。

可重复读的适用场景

可重复读,适用于对账场景,在对账过程中,其他事务对数据库的更改,不会影响校对结果。


最佳实践

  1. 基于回滚日志的描述,不建议使用长事务,长事务意味着系统中会存在很多很老的回滚日志,会占用大量存储空间

    • 另外,长事务本身可能需要非常长的时间来执行,当事务中出现问题需要回滚时,回滚本身的时间也会很长
  2. 建议保持autocomit为1,用手动开启事务的方式来使用事务。
    • begin来显示开启一个事务,mysql会自动执行set autocommit = 0;在commit或rollback后会set autocommit = 1;
    • 如果autocommit为0,每次都需要手动设置commit或rollback,不需要事务的时候也需要提交
    • autocommit为1,有些sql语句可以自动提交,防止出现长事务。
  3. 如何规避长事务
    1. 针对业务开发人员,系统培训长事务相关知识
    2. code review中,检查代码以及数据库相关配置信息
    3. 测试人员,建立相关测试用例
    4. 运维建立长事务监控脚本
  4. 长事务出现后的运维:长事务识别、处理,监控

生产项目分析

项目简介

目前的项目,使用springboot结合mybatis做数据库操作,事务管理使用Transactional注解实现,springboot默认使用的hikari pool连接池获取连接,该连接池autocommit属性值默认为true。在开启了事务的方法中,springboot会将连接的autocommit设置为false,代码如下:

类文件:org/springframework/jdbc/datasource/DataSourceTransactionManager.java

// switch to manual commit if necessary. this is very expensive in some jdbc drivers,
// so we don't want to do it unnecessarily (for example if we've explicitly
// configured the connection pool to set it already).
if (con.getautocommit()) {
txobject.setmustrestoreautocommit(true);
if (logger.isdebugenabled()) {
logger.debug("switching jdbc connection [" + con + "] to manual commit");
}
con.setautocommit(false);
}

在完成之后,如果autocommit为true,会自动恢复。

if (txObject.isMustRestoreAutoCommit()){
con.setAutoCommit(true);
}
事务失效问题

在生产实践中,很容易出现事务失效的问题,具体表现为调用事务方法,出异常后未回滚。编写代码时,按照下面的规范执行,就不会出现这个问题了。

A方法和B方法都有事务时,A方法调用B方法

  1. 如果A方法和B方法在同一个类中,需要通过容器调用B方法
  2. 如果A方法和B方法不在同一个类中,直接调用即可

A方法无事务,B方法有事务

  1. 只要A方法保证通过容器调用B方法,事务就一定会生效

具体可以参考这篇博客:

嵌套事务总结

MySQL-事务相关知识的更多相关文章

  1. Mysql 事务相关

    MySQL介绍 什么是MySQL? ​ MySQL 是一种关系型数据库,在Java企业级开发中非常常用,因为 MySQL 是开源免费的,并且方便扩展.阿里巴巴数据库系统也大量用到了 MySQL,因此它 ...

  2. mysql 索引相关知识

    由where 1 =1 引发的思考 最近工作上被说了 说代码中不能用 where 1=1,当时觉得是应该可以用的,但是找不到什么理据, 而且mysql 语句优化这方面确实很薄弱   感觉自己mysql ...

  3. mysql数据库相关知识

    什么是数据库?                数据库(Database)是按照数据结构来组织.存储和管理数据的建立在计算机存储设备上的仓库.(来自:百度) 什么是sql? 结构化查询语言(Struct ...

  4. Spring的事务管理和数据库事务相关知识

    1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱.         比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱. ...

  5. MySQL索引相关知识学习心得

    你知道的越多,你不知道的也就越多 -- 芝诺曾 一.MySQL索引学习 MySQl主要有两种类型的索引:哈希索引.B+树索引 1.哈希索引 哈希索引可以以O(1)的时间复杂度进行查找,但是这样查找导致 ...

  6. MYSQL数据库相关知识合集

    1  MYSQL取得某一范围随机数: 关键词:RAND() [产生0~1之间的随机数] mysql> SELECT RAND( ), RAND( ), RAND( ); +----------- ...

  7. MySQL启动相关知识

    使用mysqld和mysqld_safe启动的区别 直接运行mysqld程序来启动MySQL服务的方法很少见,mysqld_safe脚本[注意:mysqld_safe只是一个脚本]会在启动MySQL服 ...

  8. 数据库(mysql)相关知识

      单表查询   排序   升序   select*from表名 order by字段 asc;   降序   select*from表名 order by字段 desc;   条件查询(包括通配符) ...

  9. 二进制mysql安装相关知识

    建议安装5.x版本 高版本没安装经验的慎用 1.1 关闭防火墙systemctl stop firewalld.service #停止firewall#慎用 systemctl disable fir ...

  10. 随笔编号-06 MYSQL数据库相关知识合集

    1  MYSQL取得某一范围随机数: 关键词:RAND() [产生0~1之间的随机数] mysql> SELECT RAND( ), RAND( ), RAND( ); +----------- ...

随机推荐

  1. 使用R语言查询某物种所有通路及通路内的基因

    使用R语言查询某物种所有通路及通路内的基因,这里使用Y书的clusterProfiler包. 这里以人类为例查询所有通路及通路内的基因: library(R.utils) R.utils::setOp ...

  2. 配置Ubuntu上的NFS

    $sudo apt-get install nfs-kernel-server nfs-common 配置 $sudo vim /etc/exports#添加#/home/pi/project/roo ...

  3. BOF编写-修改时间戳

    模板配置 跟着网上的教程使用evilashz师傅的模板,下载模板解压至vs的模板目录: %UserProfile%\Documents\Visual Studio 2022\Templates\Pro ...

  4. Eval-Expression.NET:动态执行C#脚本,类似Javascript的Eval函数功能

    我们都知道在JavaScript中,我们可以通过Eval来执行JavaScript字符串代码. 下面推荐一个.Net版本的Eval的开源项目. 01 项目简介 Eval-Expression.NET是 ...

  5. C 简答题

    1.从C语⾔执⾏效率⽅便,简述下C语⾔采取了哪些措施提⾼执⾏效率.(14分 or 20分)(年年考,⾮常重要) ①使⽤指针:有些程序⽤其他语⾔也可以实现,但C能够更有效地实现:有些程序⽆法⽤其它语⾔实 ...

  6. 浅谈HPC中的Lustre

    本文分享自天翼云开发者社区<浅谈HPC中的Lustre>,作者:n****m 1. 什么是 lustre? Lustre 体系结构是一个为集群设计的存储体系结构. 其核心组件是运行在 Li ...

  7. Flask+flask-socketio+jsonrpc组合配置避坑

    Flask+flask-socketIO+jsonrpc这种组合能被我套出来也是离谱,事先声明:出现这种组合是因为本人之前对flask框架的使用仅限于flask+jsonrpc,所以导致这种情况出现, ...

  8. linux ubuntu更改软件源

    更换步骤 sudo cp /etc/apt/sources.list /etc/apt/sources.list.back sudo vim /etc/apt/sources.list 替换为下面内容 ...

  9. 【忍者算法】从生活场景理解链表反转:最重要的基础算法|LeetCode第206题 反转链表

    从生活场景理解链表反转:最重要的基础算法 为什么这道题如此重要 反转链表看似简单,却是链表操作的基石.就像建房子要先打好地基,做复杂的链表操作前必须深刻理解反转原理.无数高频面试题都建立在这个基础之上 ...

  10. [HNOI2009] 图的同构计数

    因为要求本质不同的图,容易想到群论. 为了方便处理,将边是否存在转化为边的黑白染色问题(实际上就是 \([SHOI2006]\) 有色图 的弱化版本,最终公式也差不多). 根据 \(Burnside\ ...