热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

事务隔离机制原理分析以及是否可以防止订单超卖

事务的隔离机制是指:ReadUncommitted(读取未提交内容)ReadCommitted(读取提交内容)

事务的隔离机制是指:

Read Uncommitted(读取未提交内容)
Read Committed(读取提交内容)
Repeatable Read(可重读)
Serializable(可串行化)

具体的解释最经典的MySQL书《高性能MySQL(第3版)》已经有了就不在其他地方再引用了:


隔离机制的比较



其实也有人喜欢用锁来控制并发,书中还提到了“隐式”和“显示锁定”,是这么建议的:



虽然这样,但是其实如果不经过实际的演练还是很难理解上面说的事务隔离机制到底怎么样可以防止并发。

1.查看MySQL版本


我们的版本是5.1.7


2.查看存储引擎

>show engines;


存储引擎是:InnoDB


3.实验表

假设有个商品表g,关键字段num表示库存,name表示商品名称


主要就是看不同事务隔离机制下并发修改库存是否会出现超卖。

假设我们的程序需要先查询库存,如果库存>0都可以卖,update扣库存,否则rollback。

为了制造并发肯定需要2个事务,假设是A和B。


4.确认事务隔离机制

修改会话的事务隔离级别

set session transaction isolation level read uncommitted;
set session transaction isolation level read committed;
set session transaction isolation level repeatable read;
set session transaction isolation level serializable;

>select @@global.tx_isolation,@@tx_isolation;



5.Serializable

场景一:


显然一开始AB查询的数据是一样的num=1


A开始update


这时候在等待,无法update。



过一会就超时了。


如果这个时候B也update那么一样会等待超时


所以这样,AB就会都超时。


这时即使commit也是返回0,数据库不会变化。



场景二:

A在update等待的时候,B马上commit,但是B没有update



查看结果


这次A成功的扣库存。


所以从上面可以得出一个结论:serializable是可以很好的控制并发。

然后需要把库存改为1,便于测试。


6.read committed

>set session transaction isolation level read committed;

>select @@global.tx_isolation,@@tx_isolation;



场景三:

初始化AB查出来的库存都是1,然后A可以update一条数据,无等待。



这时候AB再比较下库存,A已经是0,B是1,因为A没有commit。



然后A执行commit操作,这时候B再查已经是库存0;



这时候B执行update返回是0行,因为update不能满足where条件,所以B只有Commit,然后重新提交。



场景四:

一开始AB都是一样的库存1,然后A开始update,然后A的库存是0,B是1,因为A还没有提交。



这时候B再update



按照前面的经验,B等待其实是再等A提交,A如果一直不提交,B就会超时。



这时A提交commit,B查询就得到A更新后的结果,这时B查到库存是0自然不会去更新,也就只能结束事务。


场景五:

AB先后update,然后A在B超时之前commit,这时由于B已经读到A更新后的结果0,所以B就不能成功update。



7.repeatable read

>set session transaction isolation level repeatable read;

>select @@global.tx_isolation,@@tx_isolation;



场景六:

然后A开始update,然后A和B分别读到库存是1和0



然后A提交commit,这时候再查看A和B的库存还是保持不变。



这时候B再次尝试update


依然是返回0条,说明更新不成功。


场景八:

AB同时update


如果A不及时commit那么B肯定会超时


场景九:

就是场景八A及时commit


如果A及时commit


所以可以看出无论是read committed还是repeatable read只要update的条件where  num>0足够充分都是可以控制并发防止超卖的。

如果没有带where  num>0这个控制条件,那么肯定会可以update成功的。


8.read uncommitted

这个是需要杜绝的,就不讨论了。


9.如果没有带where  num>0,那么会怎么样呢。其实只要理解了上述流程就可以想明白会怎么样。

对于read committed

A已经update,B读到库存是0自然不会去更新;

A没有update,B读到库存是1,这要看A会不会及时提交;



如果A及时提交,B自然会去更新因为满足where条件,且成功,这样就超卖-1;



这时候由于B没有提交,所以AB分别查出0和-1


然后B提交commit,AB查出的都是-1,就不演示了。


修改会话为repeatable read

AB先后update,B在等待



然后A立即提交commit,B马上update得到返回。


结果就是-1产生了超卖:


总结:

1.使用serializable是可以防止超卖,但是性能怎么样需要数据说明;

2.read committed和repeatable read带上where条件库存num>0都是可以防止超卖的,不过需要处理超时。

3.其他各种组合情况还会更复杂,具体具体问题具体分析。


推荐阅读
  • MySQL 8.0 中的二进制日志格式详细解析及其官方文档参考。本文介绍了MySQL服务器如何使用不同的日志记录格式来记录二进制日志,包括早期版本中基于SQL语句的复制机制(即基于语句的日志记录)。此外,还探讨了其他日志记录方式,如基于行的日志记录和混合日志记录模式,并提供了配置和管理这些日志格式的最佳实践。 ... [详细]
  • 深入解析MySQL Replication中的并行复制机制与实例应用【MySQL进阶教程】
    本文深入探讨了MySQL 5.6版本后引入的并行复制机制,详细解析了其工作原理及优化效果。通过具体实例,展示了如何在实际环境中配置和使用并行复制,以提高数据同步效率和系统性能。 ... [详细]
  • 构建Java自定义持久层框架:实现数据访问与存储的高效解决方案
    JDBC连接数据库步骤:Connection连接对象,PreparedStatement ... [详细]
  • 程序连接MySQL数据库的多种方法详解 ... [详细]
  • 在编写SQL查询时,常遇到某些语句无法调用别名的问题。这主要是因为SQL和MySQL中的别名机制存在差异所致。为避免类似错误再次发生,本文汇总了相关技术资料,详细解析了别名调用的限制及其背后的原理,提供了实用的解决方案和最佳实践建议。 ... [详细]
  • Mysql进阶学习(三)排序查询与常见函数
    Mysql进阶学习(三)排序查询与常见函数一、进阶3:排序查询1、语法:2、特点:3、排序方式3.1、按单个字 ... [详细]
  • 在使用 PHP 通过 SSL 安全连接到 MySQLi 数据库服务器时,遇到了一些技术难题。我的环境包括一个 Web 服务器和一个数据库服务器,两者均使用 OpenSSL 生成了证书。尽管证书内容一致,但在尝试从 Web 服务器使用 `mysql` 命令进行连接时,仍然遇到了问题。为了确保连接的安全性和稳定性,需要进一步检查证书配置和 PHP 的 SSL 设置,以排除潜在的配置错误或兼容性问题。 ... [详细]
  • Oracle培训(三十七)——深入解析Hibernate第三章:实体关联关系映射详解
    在本节Oracle培训中,我们将深入探讨Hibernate第三章的内容,重点讲解实体关联关系映射的详细知识点。首先,回顾了Hibernate的基本概念和映射基础,随后详细分析了不同类型的实体关联关系,包括一对一、一对多和多对多关系的映射方法及其应用场景。通过具体的示例和代码片段,帮助读者更好地理解和掌握这些复杂的映射技术。此外,还讨论了如何优化关联关系的性能,以及常见的问题和解决方案。 ... [详细]
  • MySQL 5.6 引入了全局事务标识符(GTID)和多线程复制机制,显著提升了数据库的可靠性和性能。GTID 作为一种新的事务标识方式,确保了事务在主从节点间的一致性,避免了传统基于日志位置的复制可能出现的问题。多线程复制则通过并行处理多个复制任务,大幅提高了复制效率,特别是在大型数据库环境中表现更为突出。这些新特性不仅增强了 MySQL 的高可用性和扩展性,还为数据库管理带来了更多灵活性和便利性。 ... [详细]
  • Javajpa调用存储过程,抛出异常如下:javax.persistence.PersistenceException:org.hibernate.exception.Generi ... [详细]
  • SpringDataJPA是SpringBoot体系中约定优于配置的最佳实现,大大简化了项目中数据库的操作。从本课开始将会从JPA的由来开始讲解,什么是JPA、SpringBoot ... [详细]
  • 【高效构建全面的iOS直播应用】(美颜功能深度解析)
    本文深入探讨了如何高效构建全面的iOS直播应用,特别聚焦于美颜功能的技术实现。通过详细解析美颜算法和优化策略,帮助开发者快速掌握关键技术和实现方法,提升用户体验。适合对直播应用开发感兴趣的开发者阅读。 ... [详细]
  • 在PB数据窗口中,错误处理技术主要针对两类问题进行优化:一是由用户不当数据输入引发的错误,二是程序执行过程中因代码缺陷导致的异常。高效的应用程序设计需确保无论出现哪种类型的错误,系统都能有效应对,保证稳定性和用户体验。通过引入先进的错误检测与恢复机制,可以显著提升系统的健壮性和可靠性。 ... [详细]
  • 本文总结了 Apache Shiro 安全框架的学习体会,并通过具体应用实例进行了详细分析。Apache Shiro 是一个强大且灵活的安全框架,适用于各种应用程序的安全需求。通过对该框架的核心功能和使用方法的深入探讨,本文不仅帮助读者理解其基本概念和架构设计,还提供了实际项目中的应用示例,以便更好地掌握和运用这一工具。特别感谢开涛的博客文章,为本文提供了宝贵的参考和启发。 ... [详细]
  • Joomla!软件介绍【Joomla!概括介绍】国外相当知名的内容管理系统。【Joomla!基本介绍】Joomla!是一套在国外相当知名的内容管理系统(ContentManagem ... [详细]
author-avatar
被遗忘的孩子1995
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有