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

深入理解MySQL事务隔离级别与锁机制

深入理解MySQL事务隔离级别与锁机制-theme:geek-black深入理解MySQL事务隔离级别与锁机制ACID属性原子性(Atomicity):事务是一个原子操作

theme: geek-black

深入理解MySQL事务隔离级别与锁机制

ACID属性

  • 原子性(Atomicity):事务是一个原子操作,要么全部一起执行成功,要么一起执行失败
  • 一致性(Consistent):事务开始和结束,数据都必须保持一致
  • 隔离性(Isolation): 事务处理过程中需要保证相互隔离、独立
  • 持久性(Durale): 事务结束之后,对于数据的修改是永久保存下来的

并发事务处理带来的问题

  • 更新丢失(Lost Update) 或 脏写: 多个事务在不知道其他事务存在的情况下,同时对数据做修改,就会出现脏写
    • 最后的更新覆盖了其他事务所做的更新
  • 脏读(Dirty Reads): 当一个事务读取一条记录并对其他进行修改,但是还未提交,此时另一个事务也来读取,如果不加控制,第二个事务读取到的数据就是脏数据
    • 事务A读取到事务B已经修改但还未提交的数据
  • 不可重复度(Non-Repeatable Reads): 一个事务读取某些数据后的某个时间,再次读取以前读过的数据,却发现其读出的数据已经发生改变
    • 事务A内部相同查询语句在不同时刻读出的结果不一致,不符合隔离性
  • 幻读(Phantom Reads): 一个事务按相同的查询条件读取以前检索过的数据,却发现其他事务插入了满足其查询条件的新数据
    • 事务A读到了事务B提交的新增数据,不符合隔离性

隔离级别

脏读、不可重复读、幻读,其实都是数据库读一致性的问题,必须由数据库提供一定的事务隔离机制来解决,而MySQL默认的是课重复读

  • 读未提交:事务A更新了数据(还未提交),而事务B却能查询到事务A更新的数据
  • 读已提交: 事务A更新了数据(还未提交),事务B不能查询到事务A更新的数据(解决了脏读问题)
  • 可重复度: 事务A查询了记录(假如400),事务B更改了数据(400 - 50 = 350,然后commit),事务A在没有提交事务之前再次读取的值还是400,直到事务A提交事务才能读到最新更新的数据(保证了事务间的隔离性)
  • 串行化: 加了行锁的数据,其他事务不能进行增删改查等操作,相当于独占锁

锁机制

  • 锁是计算机协调多个进程/线程并发访问某一资源的机制
  • 锁分类
    • 从对数据库上的性能
      • 乐观锁(版本号对比)
      • 悲观锁
    • 从对数据库操作的类型
      • 读锁(共享锁,多个读操作可以同时进行互不影响)
      • 写锁(排它锁,写操作没完成前,其它都无法进行操作,保证串行)
    • 从堆数据库操作的粒度
      • 表锁
      • 行锁

表锁

  • 每次操作锁住整张表,开销小,加锁快,不会出现死锁,锁粒度大,发生锁冲突的概率更高,并发读最低,一般用于数据迁移的场景

行锁

  • 每次操作锁住一行数据,开销大,加锁慢,会出现死锁,锁粒度最小,发生锁冲突的概率最低,InnoDB支持事务以及行锁,但是MylSAM不支持
  • 一个session开启事务更新不提交,另外一个session更新同一条记录会阻塞,更新不同记录不会阻塞

间隙锁

  • 锁的就是两个值之间的空隙.
  • 间隙锁在某些情况下可以解决幻读问题
  • 间隙锁在可重复度隔离级别下才会生效
update account set name = '100' where id > 8 and id <18;
其他Session没法在这个范围所包含的所有行记录以及范围内间隙行记录锁在原表的间隙里插入或修改任何数据.即id在(3,20]区间都无法修改数据,注意最后那个20也是包含在内的。
  • 例如SQL: : 间隙有 id 为 (3,10),(10,20),(20,正无穷) 这三个区间.

临间锁

  • 行锁与间隙锁的组合
  • 临键锁主要母的也是为了避免幻读,如果事务降级为读已提交,临键锁会失效
  • 例如上述间隙锁例子SQL,间隙为(3,20]整个区间

无锁引行锁会升级为表锁

锁主要是加在索引上,如果对非锁引字段更新,行锁有可能会变成表锁

例如: 假如现在将无索引的char类型字段更改成另外一个varchar类型,因为没有走索引,此时会全表扫描,最终升级为'表锁'

锁优化建议

  • 尽量让所有数据检索都通过索引来完成,避免无索引行锁升级为表锁
  • 合理设计索引,尽量减小锁的范围
  • 尽可能减少检索条件范围,避免间隙锁
  • 尽量控制事务大小,减少锁定资源量和时间长度,涉及事务加锁的SQL尽量放在事务最后执行
  • 尽可能低级别事务隔离

锁小结

  • 行锁、间隙锁、临键锁都属于排它锁
  • 行锁就是锁住一行记录
  • 间隙锁只在RR级别中才会产生
  • 唯一索引只有锁住多条记录或一条不存在的记录才会产生间隙,指定给某条存在的记录加锁的时候,只会加行锁,不会产生间隙锁
  • 普通索引不管是锁住单行还是多行记录,都会产生间隙锁
  • 间隙锁锁住的两值之间的空白区域,为了防止出现幻读
  • RC情况下,间隙锁会失效
  • MylSAM在执行查询语句select前,会自动给涉及的所有表加读锁,在执行update、insert、delete等操作会自动给涉及的表加写锁.
  • InnoDB在执行查询语句select时(非串行隔离级别),不会加锁,但是update、insert、delete等操作会加行锁
  • 读锁会阻塞写,但是不会阻塞读
  • 写锁会阻塞读写
  • InnoDB存储引擎实现了行锁,虽然在锁机制的实现方面带来的性能损耗会比表锁更高,但是整体并发处理能力远远优于Mylsam的表锁锁定,所以在并发场景,InnoDB整体性能由于MylSAM
  • 如果一条SQL中用不到索引是不会使用行锁的,会使用表锁

推荐阅读
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 在Oracle11g以前版本中的的DataGuard物理备用数据库,可以以只读的方式打开数据库,但此时MediaRecovery利用日志进行数据同步的过 ... [详细]
  • Java String与StringBuffer的区别及其应用场景
    本文主要介绍了Java中String和StringBuffer的区别,String是不可变的,而StringBuffer是可变的。StringBuffer在进行字符串处理时不生成新的对象,内存使用上要优于String类。因此,在需要频繁对字符串进行修改的情况下,使用StringBuffer更加适合。同时,文章还介绍了String和StringBuffer的应用场景。 ... [详细]
  • 高质量SQL书写的30条建议
    本文提供了30条关于优化SQL的建议,包括避免使用select *,使用具体字段,以及使用limit 1等。这些建议是基于实际开发经验总结出来的,旨在帮助读者优化SQL查询。 ... [详细]
  • 本文讨论了在数据库打开和关闭状态下,重新命名或移动数据文件和日志文件的情况。针对性能和维护原因,需要将数据库文件移动到不同的磁盘上或重新分配到新的磁盘上的情况,以及在操作系统级别移动或重命名数据文件但未在数据库层进行重命名导致报错的情况。通过三个方面进行讨论。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • web.py开发web 第八章 Formalchemy 服务端验证方法
    本文介绍了在web.py开发中使用Formalchemy进行服务端表单数据验证的方法。以User表单为例,详细说明了对各字段的验证要求,包括必填、长度限制、唯一性等。同时介绍了如何自定义验证方法来实现验证唯一性和两个密码是否相等的功能。该文提供了相关代码示例。 ... [详细]
  • MySQL语句大全:创建、授权、查询、修改等【MySQL】的使用方法详解
    本文详细介绍了MySQL语句的使用方法,包括创建用户、授权、查询、修改等操作。通过连接MySQL数据库,可以使用命令创建用户,并指定该用户在哪个主机上可以登录。同时,还可以设置用户的登录密码。通过本文,您可以全面了解MySQL语句的使用方法。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • 基于事件驱动的并发编程及其消息通信机制的同步与异步、阻塞与非阻塞、IO模型的分类
    本文介绍了基于事件驱动的并发编程中的消息通信机制,包括同步和异步的概念及其区别,阻塞和非阻塞的状态,以及IO模型的分类。同步阻塞IO、同步非阻塞IO、异步阻塞IO和异步非阻塞IO等不同的IO模型被详细解释。这些概念和模型对于理解并发编程中的消息通信和IO操作具有重要意义。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • ALTERTABLE通过更改、添加、除去列和约束,或者通过启用或禁用约束和触发器来更改表的定义。语法ALTERTABLEtable{[ALTERCOLUMNcolu ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
author-avatar
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有