事务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. 解决layer在移动端关闭按钮显示一半的问题

    问题描述 layer弹出iframe,如果设置title为false,会自动设置closeBtn为2,也就是向右上方偏移了-28px,如果显示区域大于1100则正常,如果小于1100则会添加一段css ...

  2. 前端面试100-copy

    1.一些开放性题目 1.自我介绍:除了基本个人信息以外,面试官更想听的是你与众不同的地方和你的优势. 2.项目介绍 3.如何看待前端开发? 4.平时是如何学习前端开发的? 5.未来三到五年的规划是怎样 ...

  3. CPU 性能优化总结

    在Linux系统中,由于成本的限制,往往会存在资源上的不足,例如 CPU.内存.网络.IO 性能.本文,就对 Linux 进程和 CPU 的原理进行分析,总结出 CPU 性能优化的方法. 1. 分析手 ...

  4. 新型冠状病毒全国疫情API接口

    一.新浪提供的接口地址:https://interface.sina.cn/news/wap/fymap2020_data.d.json 二.163提供的接口地址:官方展示: https://news ...

  5. MongoDB:【索引类型】单字段索引、复合索引、其他索引

    查询性能分析:

  6. Docker的启停与配置等

    Docker测试题 一.选择题(每题5分) 1.关于Docker 安装的表述错误的是(C) A.Docker支持在Windows.Linux.MacOS等系统上安装 B.CentOS安装Docker有 ...

  7. 异常try-catch-finally与存储和JSON.parse

    捕获异常 捕获异常:处理可能出现的异常,当发生错误后,我们对它进行处理,不让程序崩溃. 异常处理 try-catch-finally try{ // 可能出现异常的:代码1 }catch(err){ ...

  8. 【译】轻松评估 AI 应用程序的质量

    原文 | Wendy Breiding 翻译 | 郑子铭 在构建利用 AI 的应用程序时,能够有效地评估 SLM(小型语言模型)或 LLM(大型语言模型)的响应从未如此重要. 评估是指评估 AI 模型 ...

  9. 从存钱罐到子数组:一个关于累加和的精妙问题|LeetCode 560 和为K的子数组

    LeetCode 560 和为K的子数组 点此看全部题解 LeetCode必刷100题:一份来自面试官的算法地图(题解持续更新中) 生活中的算法 你有没有这样的经历:每天往存钱罐里存一些零钱,某一天突 ...

  10. API网关-APISIX简介

    本文分享自天翼云开发者社区<API网关-APISIX简介>,作者:w****n Apache APISIX 是一个动态.实时.高性能的云原生 API 网关,提供了负载均衡.动态上游.灰度发 ...