首頁 >資料庫 >mysql教程 >你知道MySQL鎖定與事物隔離等級嗎?

你知道MySQL鎖定與事物隔離等級嗎?

coldplay.xixi
coldplay.xixi轉載
2020-10-06 14:52:232039瀏覽

你知道MySQL鎖定與事物隔離等級嗎?

相關免費學習推薦:mysql資料庫(影片)

前言

  • MySQL索引底層資料結構與演算法
  • MySQL效能最佳化原理-前篇
  • MySQL效能最佳化-實作篇1
  • MySQL效能最佳化-實務篇

##前面我們講了MySQL資料庫底層的資料結構與演算法、MySQL效能優化篇一些內容。我們再來聊聊MySQL的鎖與事務隔離級別,分上下兩篇,這篇重點講MySQL的行鎖與事務隔離級別。

鎖定定義

鎖定是電腦協調多個行程或執行緒並發存取某一資源的機制。

在資料庫中,除了傳統的運算資源(如CPU、RAM、I/O等)的爭用以外,資料也是一種供需要使用者共享的資源。如何保證資料並發存取的一致性、有效性是所有資料庫必須解決的問題,鎖定衝突也是影響資料庫並發存取效能的重要因素。

鎖定分類

  • 從效能分為樂觀鎖定(以版本比較來實現)和悲觀鎖定
  • 從資料庫操作類型分為:讀取鎖定寫入鎖定(都屬於悲觀鎖定)
    • 讀取鎖定(共享鎖定):針對同一份數據,多個讀取操作可以同時進行而不會互相影響;
    • 寫鎖(排它鎖):當前寫入操作沒有完成之前,它會阻斷其它寫鎖和讀鎖。
  • 從資料庫操作的粒度分為:表格鎖定行鎖定

對於鎖定深入的理解,可以查看《關於Java中鎖的理解》。

MySQL的鎖定

  • #(Record Locks)
  • 間隙鎖定(Gap Locks)
  • 臨鍵鎖定(Next-key Locks)
  • #共享鎖定/排他鎖定(Shared and Exclusive Locks)
  • #意向共享鎖定/意向排他鎖(Intention Shared and Exclusive Locks)
  • 插入意向鎖定(Insert Intention Locks)

  • 自增鎖(Auto-inc Locks)
預測鎖定

,這種鎖定主要用於儲存了空間資料的

空間索引。

下篇來分別聊聊,本篇重點是行鎖定以及交易隔離層級。
  • 表格鎖定
    每次操作都鎖住整張表。
開銷小,加鎖快;

不會出現死鎖;
你知道MySQL鎖定與事物隔離等級嗎?鎖定粒度大,發生鎖定衝突的機率最高;
#並發度最低。

基本操作

範例表,如下:
# 建表SQLCREATE TABLE mylock (    id INT(11) NOT NULL AUTO_INCREMENT,    NAME VARCHAR(20) DEFAULT NULL,
    PRIMARY KEY(id)
) ENGINE = MyISAM DEFAULT CHARSET = utf8;

# 插入数据INSERT INTO`test`.`mylock`(`id`,`NAME`) VALUES ('1','a'); 
INSERT INTO`test`.`mylock`(`id`,`NAME`) VALUES ('2','b'); 
INSERT INTO`test`.`mylock`(`id`,`NAME`) VALUES ('3','c'); 
INSERT INTO`test`.`mylock`(`id`,`NAME`) VALUES ('4','d');复制代码
你知道MySQL鎖定與事物隔離等級嗎?
#手動增加表格鎖定
lock table 表名称 read(write), 表名称2 read(write);复制代码

查看表上加過的鎖定
你知道MySQL鎖定與事物隔離等級嗎?
show open tables;复制代码
刪除表格鎖定

unlock tables;复制代码
案例分析— 加讀鎖定

LOCK TABLE mylock read;复制代码

    #當前session 和其他seesion 都可以讀取該表;
  • 當前session 中插入或更新鎖定表都會報錯,其他session 插入或更新則會等待。

案例分析— 加寫入鎖定

LOCK TABLE mylock WRITE;复制代码

#########當前session 對該表的增刪改查都沒有問題,其他session 對該表的所有操作都會被阻塞。 ######案例結論#########MyISAM### 在執行查詢語句(SELECT)前,會自動為涉及的所有表加讀鎖定;在執行增刪改查操作前,會自動為涉及的表加寫鎖。 #########對 MyISAM 表的讀取操作(加讀鎖定),不會阻塞其他進程同一表的讀取請求,但會阻塞對相同表的寫入請求。只有當讀鎖釋放後,才會執行其他程序的寫入操作。 ######對 MyISAM 表的寫入操作(加寫鎖定),會阻塞其他進程對相同表的讀取和寫入操作,只有當寫入鎖定釋放後,才會執行其他進程的讀寫操作。 #########總結:###讀鎖定會阻塞寫,但不會阻塞讀取;而寫鎖定則會把讀和寫都阻塞###。 ######行鎖定######每次操作都鎖住一行資料。 ###
  • 开销大,加锁慢;
  • 会出现死锁;
  • 锁定粒度最小,发生锁冲突的概率最低;
  • 并发度最高。

InnoDB 和 MyISAM 的最大不同点:

  • 支持事务(TRANSACTION)
  • 支持行级锁

行锁支持事务

事务(Transaction)及其 ACID 属性

事务是由一组 SQL 语句组成的逻辑处理单元,事务具有以下四个属性,通常简称为事务的 ACID属性

  • 原子性(Atomicity):事务是一个原子操作单元,其对数据的修改,要么全部执行,要么全部不执行。
  • 一致性(Consistent):在事务开始和完成时,数据都必须保持一致状态。这意味着所有相关的数据规则都必须应用于事务的修改,以保持数据的完整性;事务结束时,所有的内部数据结构(如B+树索引或双向链表)也都必须是正确的。
  • 隔离性(Lsolation):数据库系统提供一定的隔离机制,保障事务在不受外部并发操作影响的“独立”环境执行。这意味着事务处理过程中的中间状态对外部是不可见的,反之亦然。
  • 持久性(Durable):事务完成之后,它对于数据的修改是永久性的,即使出现系统故障也能保持。

并发事务处理带来的问题

  • 更新丢失(Lost Update)

当两个或多个事务选择同一行,然后基于最初选定的值更新该行值,由于每个事务都不知道其他事务的存在,就会发生丢失更新问题,最后的更新覆盖来其他事务所做的更新。

  • 脏读(Dirty Reads)

一个事务正在对一条记录做修改,在这个事务完成并提交前,这个条记录的数据就处于不一致的状态;这时另外一个事务也来读取同一条记录,如果不加控制,第二个事务读取来这些“脏”数据,并据此做进一步的处理,就会产生未提交的数据依赖关系。这种现象被形象的叫做“脏读”。

总结:事务A读取到来事务B已经修改但尚未提交的数据,还在这个数据基础上做来操作。此时,如果事务B回滚,事务A读取的数据无效,不符合一致性要求。

  • 不可重复读(Non-Repeatable Reads)

一个事务在读取某些数据后的某个时间,再次读取以前读过的数据,却发现其读出的数据已经发生来改变、或某些记录已经被删除了,这种现象就叫做“不可重复读”。

总结:事务A读取到了事务B已经提交的修改数据,不符合隔离性。

  • 幻读(Phantom Reads)

一个事务按相同的查询条件重新读取以前检索过的数据,却发现其他事务插入了满足其查询条件的新数据,这种现象就称为“幻读”。

总结:事务A读取到了事务B提交的新增数据,不符合隔离性。

事务隔离级别

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

你知道MySQL鎖定與事物隔離等級嗎?

数据库的事务隔离越严格,并发副作用越小,但付出的代价也就越大,因为事务隔离实质上就是使事务在一定程度上“串行化”进行,这显然与“并发”是矛盾的。

同时,不同应用对读一致性和事务隔离程度的要求也是不同的,比如许多应用对“不可重复读”和“幻读” 并不敏感,可能更关系数据并发访问的能力。

查看当前数据库的事务隔离级别

show variables like 'tx_isolation';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

设置事务隔离级别

set tx_isolation='REPEATABLE-READ';复制代码

数据库版本是5.7,隔离级别是Repeatable-Read(可重复读),不同的数据库版本和隔离级别对语句的执行结果影响很大。所以需要说明版本和隔离级别

行锁与隔离级别案例分析

事务控制语句

  • BEGINSTART TRANSACTION;显式地开启一个事务;
  • COMMIT;也可以使用 COMMIT WORK,不过二者是等价的。COMMIT会提交事务,并使已对数据库进行的所有修改称为永久性的;
  • ROLLBACK;有可以使用 ROLLBACK WORK,不过二者是等价的。回滚会结束用户的事务,并撤销正在进行的所有未提交的修改;
  • SAVEPOINT identifier;SAVEPOINT允许在事务中创建一个保存点,一个事务中可以有多个SAVEPOINT;
  • RELEASE SAVEPOINT identifier;删除一个事务的保存点,当没有指定的保存点时,执行该语句会抛出一个异常;
  • ROLLBACK TO identifier;把事务回滚到标记点;
  • SET TRANSACTION;用来设置事务的隔离级别。InnoDB存储引擎提供事务的隔离级别有READ UNCOMMITTEDREAD COMMITTEDREPEATABLE READSERIALIZABLE

事务处理方法

MYSQL 事务处理主要有两种方法:

  1. BEGIN, ROLLBACK, COMMIT来实现
    • BEGIN 开始一个事务
    • ROLLBACK 事务回滚
    • COMMIT 事务确认
  1. 直接用 SET 来改变 MySQL 的自动提交模式:
    • SET AUTOCOMMIT=0 禁止自动提交
    • SET AUTOCOMMIT=1`` 开启自动提交

示例表,如下:

CREATE TABLE `user` (    `id` INT (11) NOT NULL AUTO_INCREMENT,    `name` VARCHAR (255) DEFAULT NULL,    `balance` INT (11) DEFAULT NULL,
    PRIMARY KEY (`id`)
) ENGINE = INNODB DEFAULT CHARSET = utf8;INSERT INTO `test`.`user` (`name`,`balance`) VALUES ('zhangsan','450');INSERT INTO `test`.`user` (`name`,`balance`) VALUES ('lisi', '16000');INSERT INTO `test`.`user` (`name`,`balance`) VALUES ('wangwu','2400');复制代码

行锁演示

一个 session 开启事务更新不提交,另一个 seesion 更新同一条记录会阻塞,更新不同记录u会阻塞。

你知道MySQL鎖定與事物隔離等級嗎?
你知道MySQL鎖定與事物隔離等級嗎?

读未提交

(1)打开一个客户端A,并设置当前事务模式为 read uncommitted (读未提交),查询表 user 的初始化值

set tx_isolation='read-uncommitted';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

(2)在客户端A的事务提交之前,打开另一个客户端B,更新表 user

你知道MySQL鎖定與事物隔離等級嗎?

(3)这时,虽然客户端B的事务还没提交,但是在客户端A就可以查询到B已经更新的数据

你知道MySQL鎖定與事物隔離等級嗎?

(4)一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那么客户端A查询到的数据其实就是脏数据

你知道MySQL鎖定與事物隔離等級嗎?

(5)在客户端A执行更新语句 update user set balance = balance - 50 where id = 1; zhangsan 的 balance没有变成350,居然是400,是不是很奇怪,数据不一致啊。如果你这么想就太天真了,在应用程序中,我们会用400-50=350,并不知道其他会话回滚了,要想解决这个问题可以采用读已提交的隔离级别。

你知道MySQL鎖定與事物隔離等級嗎?

读已提交

(1)打开一个客户端A,并设置当前事务模式为 read committed (读已提交),查询表 user 的所有记录

set tx_isolation='read-committed';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

(2)在客户端A的事务提交之前,打开另一个客户端B,更新表 user

你知道MySQL鎖定與事物隔離等級嗎?

(3)这时,客户端B的事务还没提交,客户端A不能查询到B已经更新的数据,解决了脏读问题。

你知道MySQL鎖定與事物隔離等級嗎?

(4)客户端B的事务提交

你知道MySQL鎖定與事物隔離等級嗎?

(5)客户端A执行与上一步相同的查询,结果与上一步不一致,即产生了不可重复读的问题。

你知道MySQL鎖定與事物隔離等級嗎?

可重复读

(1)打开一个客户端A,并设置当前的事务模式为 repeatable read ,查询表 user 的所有记录。

set tx_isolation='REPEATABLE-READ';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

(2)在客户端A的事务提交之前,打开另一个客户端B,更新表 user 并提交。

你知道MySQL鎖定與事物隔離等級嗎?

(3)在客户端A查询表 user 的所有记录,与步骤(1)查询结果一直,没有出现不可重复读的问题。

你知道MySQL鎖定與事物隔離等級嗎?

(4)在客户端A,接着执行 update user set balance = balance - 50 where id = 1 , balance 没有变成 400 - 50 = 350, zhangsan 的 balance 的值用的是步骤(2) 中的 350 来计算的,所以是300,数据的一致性倒是没有被破坏。可重复读的隔离级别下使用了 MVCC(multi-version concurrency control)机制,select 操作不会更新版本号,是快照读(历史版本);insert、update、delete 会更新版本号,是当前读(当前版本)。

我们下篇来讲 MVCC。

你知道MySQL鎖定與事物隔離等級嗎?

(5)重新打开客户端B,插入一条新数据后提交。

你知道MySQL鎖定與事物隔離等級嗎?

(6)在客户端A查询表user 的所有记录,没有查出新增数据,所以没有出现幻读。

你知道MySQL鎖定與事物隔離等級嗎?

(7)验证幻读 在客户端A执行 update user set balance = 8888 where id = 4; ,能更新成功,再次查询到客户端B新增的数据。

串行化

(1)打开一个客户端A,并设置当前事务模式为 serializable ,查询表 user 的初始值

set tx_isolation='serializable';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

(2)打开一个客户端B,并设置当前事务模式为 serializable ,插入一条记录报错,表被锁了插入失败,MySQL 中事务隔离级别为 serializable 时会锁表,因此不会出现幻读的情况,这种隔离级别并发性极低,开发中很少会用到。

你知道MySQL鎖定與事物隔離等級嗎?

案例结论

InnoDB 存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会更高一下,但是在整体并发处理能力方面要远远优于 MyISAM 的表级锁定的。当系统并发量最高的时候,InnoDB 的整体性能和 MyISAM 相比就会有比较明显的优势。

但是,InnoDB 的行级锁定同样也有其脆弱的一面,当我们使用不当的时候,可能会让 InnoDB 的整体性能表现不仅不能比 MyISAM 高,甚至可能会更差。

行锁分析

通过检查 innodb_row_lock 状态变量来分析系统上的行锁的竞争情况:

show status like 'innodb_row_lock%';复制代码
你知道MySQL鎖定與事物隔離等級嗎?

对各个状态量的说明如下:

  • Innodb_row_lock_current_waits :当前正在等待锁定的数量
  • Innodb_row_lock_time :从系统启动到现在锁定总时间长度
  • Innodb_row_lock_time_avg :每次等待所花平均时间
  • Innodb_row_lock_time_max :从系统启动到现在等待最长的一次所花时间
  • Innodb_row_lock_waits :系统启动后到现在总共等待的次数

对于这5个状态变量,比较重要的主要是:

  • Innodb_row_lock_time_avg (等待平均时长)
  • Innodb_row_lock_waits (等待总次数)
  • Innodb_row_lock_time(等待总时长)

尤其是当等待次数很高,而且每次等待时长也不小的时候,我们就需要分析系统 中为什么会有如此多的等待,然后根据分析结果着手制定优化计划。

死锁

set tx_isolation='REPEATABLE-READ';复制代码
Session_1执行:select * from user where id=1 for update;
Session_2执行:select * from user where id=2 for update;
Session_1执行:select * from user where id=2 for update;
Session_2执行:select * from user where id=1 for update;复制代码

查看近期死锁日志信息:

show engine innodb status\G;复制代码

大多数情况mysql可以自动检测死锁并回滚产生死锁的那个事务,但是有些情况 mysql没法自动检测死锁

优化建议

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

问答

  1. MySQL 默认级别是 repeatable-read,有什么办法可以解决幻读妈?

间隙锁(Gap Lock)在某些情况下可以解决幻读问题,它是 Innodb 在 可重复读 提交下为解决幻读问题时引入的锁机制。要避免幻读可以用间隙锁在Session_1 下面执行 update user set name = 'hjh' where id > 10 and id  ,则其他 Session 没法在这个范围锁包含的间隙里插入或修改任何数据。

如:user 表有3条数据, id > 2 and id  会把第三条记录锁住,其他会话对则无法对第三条记录做操作。

你知道MySQL鎖定與事物隔離等級嗎?
你知道MySQL鎖定與事物隔離等級嗎?
  1. 无索引锁会升级为表锁,锁主要是加在索引上,如果对非索引字段更新,行锁可能会变变锁。

客户端A执行: update user set balance = 800 where name = 'zhangsan'; 

你知道MySQL鎖定與事物隔離等級嗎?

客户端B对该表任一行执行修改、删除操作都会阻塞

你知道MySQL鎖定與事物隔離等級嗎?

InnoDB 的行锁是针对索引加的锁,不是针对记录加的锁。并且该索引不能失效,否则都会从行锁升级为表锁。

  1. 鎖定某一行也可以用local in share mode(共享鎖定) 和for update(排它鎖定) ,例如: select * from test_innodb_lock where a = 2 for update; 這樣其他session 只能讀這行數據,修改則會被阻塞,直到鎖定行的session 提交。

以上是你知道MySQL鎖定與事物隔離等級嗎?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:juejin.im。如有侵權,請聯絡admin@php.cn刪除