《高性能MySQL》读书笔记之 MySQL锁、事务、多版本并发控制的基础知识

1.2 并发控制
  1.2.1 读写锁
    在处理并发读或写时,通过实现一个由两种类型的锁组成的锁系统来解决问题。这两种类型的锁通常被称为 共享锁(shared lock) 和 排它锁(exclusive lock),也叫读锁(read lock)和写锁(write lock)。
    读锁是共享的,或者说是不互相阻塞的。多个客户端可以在同一时刻读取同一个资源,而互不干扰。写锁则是排他的,也就是说一个写锁会阻塞其他写锁和读锁。
  1.2.2 锁粒度
    为了提高共享资源的并发性,尽量只锁定需要修改的部分数据,而不是所有数据。
    表锁(table lock)

      表锁是MySQL中最基本的锁策略,并且是开销最小的策略。进行写操作(插入、删除、更新等)先需要获得写锁,这会阻塞其他用户对该表的所有读写操作。读锁之间是不相互阻塞的。写锁有比读锁更高的优先级,因此一个写锁请求可能会被插入到读锁队列的前面(写锁可以插入到锁队列的前面,反之读锁则不能插入到写锁的前面)。
      ALTER TABLE之类的语句会使用表锁,并且会忽略存储引擎的锁机制。
    行级锁(row lock)
      行级锁可以最大限度的支持并发处理,但同时也到来了最大的锁开销。MySQL的InnoDB和XtraDB等存储引擎实现了行级锁。行级锁只在存储引擎层面实现,并且服务器层完全不了解存储引擎中的锁实现。
1.3 事务
  事务就是一组原子性的SQL查询,这组查询要么全部执行成功,要么全部执行失败。
  事务的ACID概念:
    原子性(atomicity):一个事务必需被视为一个不可分割的最小工作单元。
    一致性(consistency):
    隔离性(isolation):通常来说,一个事务所做的修改在最终提交以前,对其他事务是不可见的(也取决与隔离级别(Isolation level))。
    持久性(durability):一旦事务提交,则其所做的修改就会永久保存到数据库中。
  1.3.1 隔离级别
    SQL中定义了四种隔离级别:
      READ UNCOMMITTED(未提交读): 事务中的修改,即使没有提交,对其他事务也都是可见的,事务可以读取未提交的数据,也被称为脏读(Dirty Read),不推荐使用。
      READ COMMITTED(提交读): 一个事务开始时,只能看见已经提交了的事务所做的修改,也就是说,一个事务从开始直到提交之前,所做的任何修改对其他事务都是不可见的。
      REPEATABLE READ(可重复读): 解决了脏读的问题,保证在同一个事务中,多次读取同样的记录的结果是一致的。可重复读是MySQL的默认事务隔离级别。
      SERIALIZABLE(可串行化): 可串行化是最高的隔离级别,通过强制事务串行执行,避免了幻读的问题,但会导致大量的超时和锁争用的问题。
    幻读(Phantom Read): 当某个事务在读取某个范围内的记录时,另一个事务又在该范围插入了新的记录,当之前的事务再次读取该范围内的记录时,会产生幻行(Phantom Row)。
  1.3.2 死锁
    死锁是指两个或多个事务在同一资源上互相占用,并请求锁定对方占用的资源,从而导致恶性循环的现象。死锁的产生有双重原因:有些是因为真正的数据冲突,有些是存储引擎的实现方式导致的。
    InnoDB存储引擎,能检测到死锁的循环依赖,并立即返回一个错误。InnoDB目前的处理死锁的方法是:将持有最少行级排他锁的事务回滚。死锁发生时,只有部分或者完全回滚其中一个事务,才能打破僵局。大多数情况下,只需重新执行因死锁回滚的事务即可。
1.4 多版本并发控制(MVCC)
  可以认为MVCC是行级锁的一个变种,实现了非阻塞的读操作,写操作也只锁定必要的行。
  MVCC的实现,是通过保存数据在某个时间点的快照来实现的。对于多个事务,根据事务开始时间不同,每个事务对同一张表,在同一时刻看到的数据可能是不一样的。
  InnoDB的MVCC,是通过在每行记录的后面保存两个隐藏的列来实现的。这两个列,一个保存行的创建时间,一个保存行的过期时间(或删除时间)。当然存储的并不是实际的时间值,而是系统版本号(system version number)。每开始一个事务,系统版本号都会自动递增。事务开始时刻的系统版本号会作为事务的版本号,用来和查询到的每行记录的版本号进行比较。
  REPEATABLE READ隔离级别下,MVCC具体操作方式:
    SELECT:InnoDB会根据一下两个条件检查每行记录:
      a. InnoDB只查找版本号早于当前事务版本的数据行(行的系统版本号小于或等于事务的系统版本号),这样可以保证事务读取的行,要么是在事务开始之前已经存在的,要么是事务自身插入的或者修改过的。
      b. 行的删除版本要么未定义,要么大于当前事务的版本号。这可以确保事务读取到的行,在事务开始之前未被删除。
    INSERT:
      InnoDB为新插入的每一行保存当前系统版本号作为行版本号。
    DELETE:
      InnoDB为删除的每一行保存当前系统版本号作为行删除标识。
    UPDATE:
      InnoDB为插入的一行新记录,保存当前系统版本号作为行版本号,同时保存当前系统版本号到原来的行作为行删除标识。

  MVCC只在REPEATABLE READ和READ COMMITTED两个隔离级别下工作。

时间: 2024-10-19 13:36:07

《高性能MySQL》读书笔记之 MySQL锁、事务、多版本并发控制的基础知识的相关文章

MySQL学习笔记之一 MySQL入门

本人之前接触的关系型数据库主要是oracle和sqlserver,而对于mysql知之甚少,但查阅网上资料发现,mysql与oracle非常相似,所以学起来应该不会很费劲,在总结的时候可能更多的把关注点放在它与oracle的不同之处. 一.简介 MySQL是一个真正的多用户.多线程SQL数据库服务器.SQL(结构化查询语言)是世界上最流行的和标准化的数据库语言.MySQL是一个客户端/服务器结构的实现, 它由一个服务器守护程序mysqld和很多不同的客户程序和库组成. MySQL的普及并不局限于

《高性能MySQL》读书笔记--锁、事务、隔离级别 转

1.锁 为什么需要锁?因为数据库要解决并发控制问题.在同一时刻,可能会有多个客户端对表中同一行记录进行操作,比如有的在读取该行数据,其他的尝试去删除它.为了保证数据的一致性,数据库就要对这种并发操作进行控制,因此就有了锁的概念. 1.1锁的分类 从对数据操作的类型(读\写)分 读锁(共享锁):针对同一块数据,多个读操作可以同时进行而不会互相影响. 写锁(排他锁):当前写操作没有完成前,它会阻断其他写锁和读锁. 大多数时候,MySQL锁的内部管理都是透明的. 1.2锁粒度(Lock granula

《高性能MySQL》读书笔记--锁、事务、隔离级别

1.锁 为什么需要锁?因为数据库要解决并发控制问题.在同一时刻,可能会有多个客户端对表中同一行记录进行操作,比如有的在读取该行数据,其他的尝试去删除它.为了保证数据的一致性,数据库就要对这种并发操作进行控制,因此就有了锁的概念. 1.1锁的分类 从对数据操作的类型(读\写)分 读锁(共享锁):针对同一块数据,多个读操作可以同时进行而不会互相影响. 写锁(排他锁):当前写操作没有完成前,它会阻断其他写锁和读锁. 大多数时候,MySQL锁的内部管理都是透明的. 1.2锁粒度(Lock granula

高性能MySQL --- 读书笔记(2) - 2016/8/2

第1章 MySQL架构 MySQL架构与其他数据库服务器大不相同,这使它能够适应广泛的应用.MySQL足够灵活,能适应高要求架构.例如Web应用,同时还适用于嵌入式应用.数据仓库.内容索引和分发软件.高可用的冗余系统.联机事务处理系统OLTP及很多其他应用类型. 为了充分发挥MySQL的性能,顺畅地使用它,就必须理解它的设计.MySQL的灵活性体现在很多方面,它可以再众多硬件平台上良好的配置和运行,还支持多种数据类型.不过MySQL最重要.最不同寻常的特征是它的存储引擎框架,这种架构可以讲查询处

[读书笔记]xampp mysql启动失败解析(win7)

1.  [mysql]  MySQL Service detected with wrong path  [mysql]  Change XAMPP MySQL and Control Panel settings or  [mysql]  Uninstall/disable the other service manually first  [mysql]  Found Path: "D:\Program Files\MySQL\MySQL Server 5.6.7\bin\mysqld-nt

MySQL学习笔记 初涉MySQL

1.在Linux下安装MySQL # yum -y install mysql mysql-server mysql-devel        修改字符集:/etc/my.conf配置文件 vi /etc/my.conf [mysqld] default-character-set=utf8 character_set_server=utf8 [client] default-character-set=utf8 2.启动和停止MySQL服务 # service mysqld start 3.登

高性能javascript读书笔记(三.DOM 编程1)

第三章DOM Script DOM编程 读书笔记 访问和修改DOM元素 浏览器通常要求DOM实现和JavaScript保持相互独立. <!-- 例如IE中,被称为JScript的JavaScript实现位于库文件jscript.dll中,而DOM实现位于另一个库mshtml.dll(内 部代号Trident).这种分离技术允许其他技术和语言,如VBScript,受益于Trident所提供的DOM功能和渲染功能.Safari使用Webkit的WebCore处理DOM和渲染,具有一个分离的JavaS

高性能Mysql读书笔记7.12.1

缓存存放在一个应用表中,通过一个哈希值应用,这个哈希值包括了如下因素,查询本身,当前要查询的数据库,客户端协议的版本等一些其他可能会影响返回结果的信息. 当判断缓存是否命中时,Mysql不会解析,"正规化"或者参数化查询语句,而是直接适用SQL语句和客户端发送过来的其他原始信息.任何字符上的不同,例如空额,注释等都会导致缓存的不命中,所以统一的编码规则是一个好的习惯. 检查缓存的时候并没有解析SQL语句,所以Mysql并不知道查询语句中是否包含不确定函数.在检查缓存之前,Mysql就做

高性能mysql读书笔记(一):Schema与数据类型优化

4.5 加快ALTER TABLE 操作的速度 原理: MySQL 的ALTER TABLE 操作的性能对大表来说是个大问题. MySQL 执行大部分修改表结构操作的方法是用新的结构创建一个空表,从旧表中查出所有数据插入新表,然后删除旧表.这样操作可能需要花费很长时间,如果内存不足而表又很大,而且还有很多索引的情况下尤其如此 特点: 大部分ALTER TABLE 操作将导致MySQL 服务中断 ALTER TABLE 本质是建新结构的表,从旧表插入数据到新表 (SHOW STATUS 显示这个语