MySQL-事务相关知识
事务ACID的理解
引入事务的主要目的:
- 保证数据库从一个一致性状态切换为另一种一致性状态
- 所有修改要么都保存,要么都不保存
A 原子性
原子性关注单个事务的整体性,需要保证事务中的全部操作是一个单元,要么都成功,要么都失败。
C 一致性
一致性关注,事务开始和结束后,数据库的完整性约束没有被破坏。简单理解,如果有外键的存在,事务执行前,外键可以保证一致性约束;事务执行之后,这个一致性约束也不应该被破坏。
I 隔离性
隔离性关注多个事务之间是否彼此互相不知晓,不影响。
没有隔离性带来的问题
没有隔离级别的时候,多个事务互相交替执行,会出现一些问题:
- 脏读:读取到其他事务未提交的数据
- 不可重复读:在事务的开始和结束这段时间,对同一行数据,读取到的值是不同的
- 幻读:事务开始和结束这段时间,同一个查询,查询到的数据行数不一致
四种隔离级别
四种隔离级别,可以分别解决不同的隔离性问题:
- 读未提交:事务可以读到其他事务未提交的数据
- 有脏读,不可重复读,幻读问题
- 读已提交:事务可以读到其他事务已经提交的数据
- 解决脏读问题,仍有不可重复读,幻读问题
- 可重复读:事务在开始到结束这段时间,读到的同一行数据是一致的。
- 解决脏读、不可重复读,仍有幻读问题
- 串行化:事务和事务之间是串行执行的
- 所有问题都可以解决
这四种隔离级别的隔离读,从上到下越来越高,但是性能从上到下越来越低。
隔离性的实现
实现上,数据库会创建一个视图,访问的时候以这个视图的逻辑结果为准。
- 读未提交:不创建视图
- 读已提交:在事务中的每条sql语句执行前创建视图
- 可重复读:事务开始时创建,整个事务执行过程中都已这个视图为准
- 串行化:直接使用加锁的方式实现
在每一次更新操作以后,会记录一条回滚日志,当前的最新值,可以通过回滚日志回滚到之前的某一个状态,MVCC(多版本并发控制)就是通过回滚段实现的,同一条记录在系统中存在多个版本,不同时刻启动的事务,查看到的就是不同的版本值,并且版本之间互不干扰。
回滚日志是会被删除的,当系统判断,没有事务会用到这些回滚日志的时候就会删除回滚日志,也就是说当这个系统中没有比这个回滚日志更早的事务视图的时候。
可重复读的适用场景
可重复读,适用于对账场景,在对账过程中,其他事务对数据库的更改,不会影响校对结果。
最佳实践
- 基于回滚日志的描述,不建议使用长事务,长事务意味着系统中会存在很多很老的回滚日志,会占用大量存储空间
- 另外,长事务本身可能需要非常长的时间来执行,当事务中出现问题需要回滚时,回滚本身的时间也会很长
- 建议保持autocomit为1,用手动开启事务的方式来使用事务。
- begin来显示开启一个事务,mysql会自动执行set autocommit = 0;在commit或rollback后会set autocommit = 1;
- 如果autocommit为0,每次都需要手动设置commit或rollback,不需要事务的时候也需要提交
- autocommit为1,有些sql语句可以自动提交,防止出现长事务。
- 如何规避长事务
- 针对业务开发人员,系统培训长事务相关知识
- code review中,检查代码以及数据库相关配置信息
- 测试人员,建立相关测试用例
- 运维建立长事务监控脚本
- 长事务出现后的运维:长事务识别、处理,监控
生产项目分析
项目简介
目前的项目,使用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方法
- 如果A方法和B方法在同一个类中,需要通过容器调用B方法
- 如果A方法和B方法不在同一个类中,直接调用即可
A方法无事务,B方法有事务
- 只要A方法保证通过容器调用B方法,事务就一定会生效
具体可以参考这篇博客:
嵌套事务总结
MySQL-事务相关知识的更多相关文章
- Mysql 事务相关
MySQL介绍 什么是MySQL? MySQL 是一种关系型数据库,在Java企业级开发中非常常用,因为 MySQL 是开源免费的,并且方便扩展.阿里巴巴数据库系统也大量用到了 MySQL,因此它 ...
- mysql 索引相关知识
由where 1 =1 引发的思考 最近工作上被说了 说代码中不能用 where 1=1,当时觉得是应该可以用的,但是找不到什么理据, 而且mysql 语句优化这方面确实很薄弱 感觉自己mysql ...
- mysql数据库相关知识
什么是数据库? 数据库(Database)是按照数据结构来组织.存储和管理数据的建立在计算机存储设备上的仓库.(来自:百度) 什么是sql? 结构化查询语言(Struct ...
- Spring的事务管理和数据库事务相关知识
1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱. 比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱. ...
- MySQL索引相关知识学习心得
你知道的越多,你不知道的也就越多 -- 芝诺曾 一.MySQL索引学习 MySQl主要有两种类型的索引:哈希索引.B+树索引 1.哈希索引 哈希索引可以以O(1)的时间复杂度进行查找,但是这样查找导致 ...
- MYSQL数据库相关知识合集
1 MYSQL取得某一范围随机数: 关键词:RAND() [产生0~1之间的随机数] mysql> SELECT RAND( ), RAND( ), RAND( ); +----------- ...
- MySQL启动相关知识
使用mysqld和mysqld_safe启动的区别 直接运行mysqld程序来启动MySQL服务的方法很少见,mysqld_safe脚本[注意:mysqld_safe只是一个脚本]会在启动MySQL服 ...
- 数据库(mysql)相关知识
单表查询 排序 升序 select*from表名 order by字段 asc; 降序 select*from表名 order by字段 desc; 条件查询(包括通配符) ...
- 二进制mysql安装相关知识
建议安装5.x版本 高版本没安装经验的慎用 1.1 关闭防火墙systemctl stop firewalld.service #停止firewall#慎用 systemctl disable fir ...
- 随笔编号-06 MYSQL数据库相关知识合集
1 MYSQL取得某一范围随机数: 关键词:RAND() [产生0~1之间的随机数] mysql> SELECT RAND( ), RAND( ), RAND( ); +----------- ...
随机推荐
- 使用R语言查询某物种所有通路及通路内的基因
使用R语言查询某物种所有通路及通路内的基因,这里使用Y书的clusterProfiler包. 这里以人类为例查询所有通路及通路内的基因: library(R.utils) R.utils::setOp ...
- 配置Ubuntu上的NFS
$sudo apt-get install nfs-kernel-server nfs-common 配置 $sudo vim /etc/exports#添加#/home/pi/project/roo ...
- BOF编写-修改时间戳
模板配置 跟着网上的教程使用evilashz师傅的模板,下载模板解压至vs的模板目录: %UserProfile%\Documents\Visual Studio 2022\Templates\Pro ...
- Eval-Expression.NET:动态执行C#脚本,类似Javascript的Eval函数功能
我们都知道在JavaScript中,我们可以通过Eval来执行JavaScript字符串代码. 下面推荐一个.Net版本的Eval的开源项目. 01 项目简介 Eval-Expression.NET是 ...
- C 简答题
1.从C语⾔执⾏效率⽅便,简述下C语⾔采取了哪些措施提⾼执⾏效率.(14分 or 20分)(年年考,⾮常重要) ①使⽤指针:有些程序⽤其他语⾔也可以实现,但C能够更有效地实现:有些程序⽆法⽤其它语⾔实 ...
- 浅谈HPC中的Lustre
本文分享自天翼云开发者社区<浅谈HPC中的Lustre>,作者:n****m 1. 什么是 lustre? Lustre 体系结构是一个为集群设计的存储体系结构. 其核心组件是运行在 Li ...
- Flask+flask-socketio+jsonrpc组合配置避坑
Flask+flask-socketIO+jsonrpc这种组合能被我套出来也是离谱,事先声明:出现这种组合是因为本人之前对flask框架的使用仅限于flask+jsonrpc,所以导致这种情况出现, ...
- linux ubuntu更改软件源
更换步骤 sudo cp /etc/apt/sources.list /etc/apt/sources.list.back sudo vim /etc/apt/sources.list 替换为下面内容 ...
- 【忍者算法】从生活场景理解链表反转:最重要的基础算法|LeetCode第206题 反转链表
从生活场景理解链表反转:最重要的基础算法 为什么这道题如此重要 反转链表看似简单,却是链表操作的基石.就像建房子要先打好地基,做复杂的链表操作前必须深刻理解反转原理.无数高频面试题都建立在这个基础之上 ...
- [HNOI2009] 图的同构计数
因为要求本质不同的图,容易想到群论. 为了方便处理,将边是否存在转化为边的黑白染色问题(实际上就是 \([SHOI2006]\) 有色图 的弱化版本,最终公式也差不多). 根据 \(Burnside\ ...