前言

本篇文章主要讲诉数据库中事务的四大特性(ACID)以及事务的隔离级别划分。

数据库事务及其特性

事务是指满足ACID特性的一组操作,可以通过 Commit 提交一个事务,也可以使用 Rollback 进行回滚。对于MySQL的InnoDB引擎,其和MyIsAm引擎的主要区别就是InnoDB支持事务(题外话:InnoDB是MySQL5.5以后的默认执行引擎)。下面我们来介绍下事务的四大特性:

原子性(Atomicity)

原子性是指事务包含的所有操作要么全部成功,要么全部失败回滚,所以事务的操作如果成功就必须要完全应用到数据库,如果操作失败则不能对数据库有任何影响。为了实现原子性,需要通过日志将所有对数据的更新操作都写入日志,如果一个事务中的一部分操作已经成功,但以后的操作,由于断电/系统崩溃/其它的软硬件错误而无法继续,则通过回溯日志,将已经执行成功的操作撤销,从而达到“全部操作失败”的目的。

一致性(Consistency)

一致性是指事务必须使数据库从一个正确状态变换到另一个正确的状态,也就是说一个事务执行之前和执行之后都必须处于正确的状态。举例:拿转账来说,假设用户A有1000块钱,用户B有2000块钱,那么不管A和B之间如何转账,转几次账,事务结束后两个用户的钱相加起来应该还得是3000,这就是事务的一致性。事务不一致的情况:(1)用户A给用户B转500块钱,此时用户A账户减去500元,剩下500元,但是此时突然系统崩溃了,用户B没来得及加500,此时的数据库就出现了不一致的状态。(2)事务1需要将500元转入帐号A:先读取帐号A的值,然后在这个值上加上500。但是,在这两个操作之间,另一个事务2修改了帐号A的值,为它增加了500元。那么最后的结果应该是A增加了1000元。但事实上,事务1最终完成后,帐号A只增加了500元,因为事务2的修改结果被事务1覆盖掉了。这使得三个账户转账后,数据的不一致。

隔离性(Isolation)

隔离性是当多个用户并发访问数据库时,比如操作同一张表时,数据库为每一个用户开启的事务,不能被其他事务的操作所干扰,多个并发事务之间要相互隔离。

即要达到这么一种效果:对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这样每个事务都感觉不到有其他事务在并发地执行。

关于事务的隔离性数据库提供了多种隔离级别,稍后会介绍到。

持久性(Durability)

持久性是指一个事务一旦被提交了,那么对数据库中的数据的改变就是永久性的,即便是在数据库系统遇到故障的情况下也不会丢失提交事务的操作。

例如我们在使用JDBC操作数据库时,在提交事务方法后,提示用户事务操作完成,当我们程序执行完成直到看到提示后,就可以认定事务以及正确提交,即使这时候数据库出现了问题,也必须要将我们的事务完全执行完成,否则就会造成我们看到提示事务处理完毕,但是数据库因为故障而没有执行事务的重大错误。

ACID 之间的关系

用一句话描述ACID的关系就是:原子性,隔离性和持久性都是为了保证数据库数据的一致性

隔离级别

介绍完事务的四大特性(简称ACID)后,现在重点来说明下事务的隔离性,当多个线程都开启事务操作数据库中的数据时,数据库系统要能进行隔离操作,以保证各个线程获取数据的准确性,在介绍数据库提供的各种隔离级别之前,我们先看看如果不考虑事务的隔离性,会发生的几种问题:

1. 脏读

脏读是指并发过程中,一个事务处理过程里读取了另一个未提交的事务中的数据。

当一个事务正在多次修改某个数据,而在这个事务中这多次的修改都还未提交,这时一个并发的事务来访问该数据,就会造成两个事务得到的数据不一致。例如:用户A向用户B转账100元,对应SQL命令如下:

update account set money=money+100 where name=’B’; (此时A通知B) update account set money=money - 100 where name=’A’;

当只执行第一条SQL时,A通知B查看账户,B发现确实钱已到账(此时即发生了脏读),而之后无论第二条SQL是否执行,只要该事务不提交,则所有操作都将回滚,那么当B以后再次查看账户时就会发现钱其实并没有转。

2. 不可重复读

不可重复读是指在对于数据库中的某条数据,一个事务范围内多次查询却返回了不同的数据值,这是由于在查询间隔,被另一个事务修改并提交了。例如事务T1在读取某一数据,而事务T2立马修改了这个数据并且提交事务给数据库,事务T1再次读取该数据就得到了不同的结果,发生了不可重复读。

不可重复读和脏读的区别是:

脏读是某一事务读取了另一个事务未提交的脏数据,而不可重复读则是两次读取之间存在另一个事务提交的数据。

在某些情况下,不可重复读并不是问题,比如我们多次查询某个数据当然以最后查询得到的结果为主。但在另一些情况下就有可能发生问题,例如对于同一个数据A和B依次查询就可能不同,A和B就可能打起来了……

3. 幻读

幻读是事务非独立执行时发生的一种现象。例如事务T1查询整张表中有多少条记录,这时事务T2又对这个表中插入了一行数据。而操作事务T1的用户如果再查看整张表有多少行数据,会发现多出一行数据,其实这行是事务T2添加的,就好像产生幻觉一样,这就是发生了幻读。

幻读和不可重复读区别是:

For locking reads (SELECT with FOR UPDATE or LOCK IN SHARE MODE),UPDATE, and DELETE statements, locking depends on whether the statement uses a unique index with a unique search condition, or a range-type search condition.

(引用自MySQL官方手册:https://dev.mysql.com/doc/refman/5.7/en/innodb-transaction-isolation-levels.html)

不可重复读的重点是修改(update)或者删除(delete),操作的是某一行数据,需要锁行。同样的条件, 你读取过的数据, 再次读取出来发现值不一样了。

幻读的重点在于新增(insert),操作的是整张表,需要锁表。同样的条件, 第1次和第2次读出来的记录数不一样

四种隔离级别

在数据库操作中,为了有效保证并发读取数据的正确性,提出的事务隔离级别,MySQL数据库为我们提供的四种隔离级别。

未提交读(Read Uncommitted):允许脏读,也就是可能读取到其他会话中未提交事务修改的数据

提交读(Read Committed):只能读取到已经提交的数据。Oracle等多数数据库默认都是该级别 (不重复读)

可重复读(Repeated Read):可重复读。在同一个事务内的查询都是事务开始时刻一致的,InnoDB默认级别。在SQL标准中,该隔离级别消除了不可重复读,但是还存在幻象读

串行读(Serializable):完全串行化的读,每次读都需要获得表级共享锁,读写相互都会阻塞

以上四种隔离级别最高的是Serializable级别,最低的是Read uncommitted级别,当然级别越高,执行效率就越低。像Serializable这样的级别,就是以锁表的方式(类似于Java多线程中的锁)使得其他的线程只能在锁外等待,所以平时选用何种隔离级别应该根据实际情况。InnoDB引擎默认的事务隔离级别为Repeatable read (可重复读)。

InnoDB引擎支持上面四种隔离级别,默认的为Repeatable read (可重复读);而在Oracle数据库中,只支持Serializable (串行化)级别和Read committed (读已提交)这两种级别,其中默认的为Read committed级别。

参考文献

mysql数据库1对n_MySQL数据库InnnoDB引擎事务说明相关推荐

  1. oracle、mysql、sqlserver的对比数据库引擎的对比与选型InnoDB解决幻读

    1. 所属公司: MySQL是由瑞典MySQL AB公司开发,目前属于 Oracle 旗下产品: Oracle是由美国ORACLE公司(甲骨文)开发的一组核心软件产品: SqlServer是由Micr ...

  2. mysql innodb文件存储_MySQL数据库和InnoDB存储引擎文件

    参数文件 当MySQL示例启动时,数据库会先去读一个配置参数文件,用来寻找数据库的各种文件所在位置以及指定某些初始化参数,这些参数通常定义了某种内存结构有多大等.在默认情况下,MySQL实例会按照一定 ...

  3. mysql 查找相似数据_数据库存储引擎大揭秘,不看不知道这里面的骚操作可真多!...

    吊打各种树这篇文章 带大家学习一遍数据结构中的各种树,对数据结构还不够熟悉的同学,那篇文章可以作为基础入门,我画了很多图理解起来不困难,建议回头先学习下那篇文章,更容易理解本文要讲的内容. 文章里有提 ...

  4. MySQL数据库介绍——包含索引、视图、事务、引擎

    文章目录 什么是数据库 数据库安装 索引 视图 事务 MySQL数据库优化方式 引擎 什么是数据库 平时我们大家看到的各种网站.网页,其实里面最底层的内容基本上都来自于数据库.首先我们看到的Web页面 ...

  5. mysql 的innoDB和NDB数据库引擎

    1 MySQL的innodb和cluster的NDB引擎都支持事务,在有共同的特性外,也有不同之处: 以mysql cluster NDB 7.3和MySQL 5.6之InnoDB为例: ndb7.3 ...

  6. 初识mysql数据字段属性_MySQL数据库~~~~初识、基础数据类型

    一 数据库初识 1.1 什么是数据库 数据库(DataBase,简称DB),简而言之可视为电子化的文件柜----存储电子文件的处所,用户可以对文件中的数据运行新增,截取,更新,删除等操作. 所谓数据库 ...

  7. mysql实训报告_mysql数据库技术》实验报告.doc

    mysql数据库技术>实验报告 MySQL数据库技术实验报告 系 别 班 级 学 号 姓 名 地点 地点机房 课程名称 MySQL数据库技术 实验名称 实验1 MySQL的使用 实 验 过 程 ...

  8. MySQL之父等国际数据库掌门人齐聚,1024 程序员节全体大会重磅官宣!

    10月23-24日,由CSDN.长沙市政府及多家机构联合主办的第二届"长沙·中国1024程序员节"(1024.csdn.net)将盛大举行.今年程序员节活动囊括:岳麓书院尖峰对话, ...

  9. mysql优化零基础_MySQL8数据库 | MySQL调优|MySQL底层原理|MySQL零基础新手教程

    MySQL8数据库安装 一.Windows 环境下安装 Select Operating System: Microsoft Windows B.解压并配置MySQL环境变量 MYSQL_HOME: ...

最新文章

  1. pandas创建内容全是0的dataframe、pandas基于随机整数、随机浮点数创建dataframe(random numbers)
  2. thinkphp的伪静态化
  3. C++ | 蓝桥杯2021年第十二届卡片问题(源代码:C语言/C++/Python)
  4. 一定是h的方式不对阅读_德国留学 ▏德国高速真要限速350km/h了?!
  5. 特征计算 -获取本周,本月初,本月末,上月同日,本季,本年初,本年末的日期 - (Python、MySQL、Oracle)
  6. python中ans的用法_python cx_Oracle基础使用方法
  7. Visio studio 2015企业版,汉语版下载,安装,破解,搞定了
  8. windows的盘共享挂载到linux下,通过Windows共享文件夹直接挂载到Linux中使用(实验操作)...
  9. 华为服务器本地做系统,华为服务器远程做系统安装
  10. as3通用三方库大全
  11. python_图像去畸变/图像矫正
  12. 安装jdk的时候为什么会有两个jre文件
  13. LintCode 吹气球
  14. 分享一个好的清理系统垃圾软件
  15. 企业邮箱部署SSL证书
  16. Android 类似手机接收到短信桌面图标改变
  17. 脉脉行业头条业务的思路猜想
  18. pcb规则中英文对照
  19. kotlin发音!这份字节跳动历年校招Android面试真题解析,附架构师必备技术详解
  20. 人工智能:声纹相关基础概念介绍

热门文章

  1. javascript实现分页效果
  2. 解决jQuery与其他库冲突的方法
  3. HTML-超链接锚点笔记
  4. 取到小数后三位_小数名师工作室 数学实验 || 一个小数乘10、100、1000……的计算规律...
  5. PTA8、列表偶数位置的元素操作 (10 分)
  6. Ubuntu truffle框架安装
  7. oracle++dtcol,Oracle 中的 Incarnation 到底是个什么?实验操作篇
  8. 查询数据库前十条数据_入门数据分析的一些建议
  9. Shell 07 项目案例
  10. element-UI ,Table组件实现拖拽效果