数据库一般都会并发执行多个事务,多个事务可能会并发的对相同的一批数据进行增删改查操作,可能就会导致脏写、脏读、不可重复读、幻读这些问题。为了解决多事务并发问题,数据库设计了事务隔离机制、锁机制、MVCC多版本并发控制隔离机制,用一整套机制来解决多事务并发问题。本章主要是同读者一起学习,认识感受这些概。念
事务是由一组SQL语句组成的逻辑处理单元,事务具有以下4个属性。
“脏读”、“不可重复读”和“幻读”,其实都是数据库读一致性问题,必须由数据库提供一定的事务隔离机制来解决。
隔离级别 | 脏读 | 不可重复读 | 幻读 |
---|---|---|---|
读未提交 | 可能 | 可能 | 可能 |
读已提交 | 不可能 | 可能 | 可能 |
可重复读 | 不可能 | 不可能 | 可能(MySQLInnoDB不可能) |
可串行化 | 不可能 | 不可能 | 不可能 |
数据库的事务隔离越严格,并发副作用越小,但付出的代价也就越大,因为事务隔离实质上就是使事务在一定程度上“串行化”进行,这显然与“并发”是矛盾的。不同的应用对读一致性和事务隔离程度的要求也是不同的,有的应用对“不可重复读"和“幻读”并不敏感,可能更关心数据并发访问的能力。
查看当前数据库的事务隔离级别:
show variables like 'tx_isolation';
mysql8.0,tx_isolation变量改为transaction_isolation
show variables like 'transaction_isolation'
设置事务隔离级别:
SET [SESSION | GLOBAL] TRANSACTION ISOLATION LEVEL {READ UNCOMMITTED | READ COMMITTED | REPEATABLE READ | SERIALIZABLE}
Mysql默认的事务隔离级别是可重复读,用Spring开发程序时,如果不设置隔离级别默认用Mysql设置的隔离级别。
锁是计算机协调多个进程或线程并发访问某一资源的机制。
在数据库中,除了传统的计算资源(如CPU、RAM、I/O等)的争用以外,数据也是一种供需要用户共享的资源。如何保证数据并发访问的一致性、有效性是所有数据库必须解决的一个问题,锁冲突也是影响数据库并发访问性能的一个重要因素。
从性能上分:
从对数据库操作的类型分:(都属于悲观锁)
从对数据操作的粒度分:
每次操作锁住整张表。开销小,加锁快;不会出现死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低;一般用在整表数据迁移的场景。
手动增加表锁
lock table 表名称 read(write),表名称2 read(write);
show open tables;
删除表锁
unlock tables;
每次操作锁住一行数据。开销大,加锁慢;会出现死锁;锁定粒度最小,发生锁冲突的概率最低,并发度最高。
一个session开启事务更新不提交,另一个session更新同一条记录会阻塞,更新不同记录不会阻塞
.
InnoDB与MYISAM的不同:
案例表
CREATE TABLE `account` ( `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`.`account` (`name`, `balance`) VALUES ('lilei', '450'); INSERT INTO `test`.`account` (`name`, `balance`) VALUES ('hanmei', '16000'); INSERT INTO `test`.`account` (`name`, `balance`) VALUES ('lucy', '2400');
实验开始前,先关闭自动提交
set global autocommit = 0;
打开一个客户端A,并设置当前事务模式为read uncommitted(读未提交)
set global transaction isolation level read uncommitted;
查询表account的初始值:
start transaction;-- 开启事务 select * from account
打开另一个客户端B,更新表account:
start transaction; update account set balance = balance - 50 where id = 1;
此时不能用navicat,navicat会自动提交
这时,虽然客户端B的事务还没提交,但是客户端A就可以查询到B已经更新的数据:
一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那客户端A查询到的数据其实就是脏数据。
客户端A修改隔离级别
set global transaction isolation level read committed;
客户端A开启事务查询
start transaction; select * from account;
重启新的客户端B,开启事务,修改表
update account set balance = balance - 50 where id = 1;
这时,客户端B的事务还没提交,客户端A不能查询到B已经更新的数据,解决了脏读问题:
客户端B的事务提交
commit;
客户端A执行与上一步相同的查询,结果 与上一步不一致,即产生了不可重复读的问题
客户端A,设置当前事务模式为repeatable read
set global transaction isolation level repeatable read;
查询表account的所有记录
打开另一个客户端B,更新表account并提交
start transaction; update account set balance = balance - 50 where id = 1; commit;
在客户端A查询表account的所有记录,
在客户端A,接着执行
update account set balance = balance - 50 where id = 1
balance没有变成450-50=400,lilei的balance值用的是步骤2中的400来算的,所以是350,数据的一致性倒是没有被破坏。
可重复读的隔离级别下使用了MVCC(multi-version concurrency control)机制,select操作不会更新版本号,是快照读(历史版本);insert、update和delete会更新版本号,是当前读(当前版本)。
重新打开客户端B,插入一条新数据后提交,
begin; insert into account values(4,'antry',666); commit;
随后如果在客户端A能查看到这条消息就是幻读。因为前面表格中降到了可重复读下(MySQLInnoDB不可能)出现幻读,所以这边没有幻读的效果。
但,此时在客户端A能够修改客户端新增的id为4的数据,且再次查询能查到客户端B新增的数据。因此又可以证明幻读的存在了。
update account set balance = 888 where id = 4;
客户端A,设置当前事务模式为serializable
set global transaction isolation level serializable;
查询表account的初始值:
打开一个客户端B,更新相同的id为1的记录会被阻塞等待,如果很久等不到就会报超时。
更新id为2的记录可以成功,说明在串行模式下innodb的查询也会被加上行锁。
如果客户端A执行的是一个范围查询,那么该范围内的所有行包括每行记录所在的间隙区间范围(就算该行数据还未被插入也会加锁,这种是间隙锁)都会被加锁。此时如果客户端B在该范围内插入数据都会被阻塞,所以就避免了幻读。
间隙锁,锁的就是两个值之间的空隙。Mysql默认级别是repeatable-read,间隙锁在某些情况下可以解决幻读问题。
上图数据间隙有 id 为 (3,10),(10,20),(20,正无穷) 这三个区间
在Session_1下面执行 update account set name = ‘antry’ where id > 8 and id <18;,则其他Session没法在这个范围所包含的所有行记录(包括间隙行记录)以及行记录所在的间隙里插入或修改任何数据,即id在(3,20]区间都无法修改数据,最后那个20也是包含在内的。
间隙锁是在可重复读隔离级别下才会生效。
Next-Key Locks是行锁与间隙锁的组合。像上面那个例子里的这个(3,20]的整个区间可以叫做临键锁。
锁主要是加在索引上,如果对非索引字段更新,行锁可能会变表锁
锁定某一行还可以用lock in share mode(共享锁) 和for update(排它锁),例如:select * from test_innodb_lock where a = 2 for update; 这样其他session只能读这行数据,修改则会被阻塞,直到锁定行的session提交。
Innodb存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会要更高一下,但是在整体并发处理能力方面要远远优于MYISAM的表级锁定的。当系统并发量高的时候,Innodb的整体性能和MYISAM相比就会有比较明显的优势了。 但是,Innodb的行级锁定同样也有其脆弱的一面,当我们使用不当的时候,可能会让Innodb的整体性能表现不仅不能比MYISAM高,甚至可能会更差。
通过检查InnoDB_row_lock状态变量来分析系统上的行锁的争夺情况
show status like 'innodb_row_lock%';
当等待次数很高,而且每次等待时长也不小的时候,我们就需要分析系统中为什么会有如此多的等待,然后根据分析结果着手制定优化计划。
查看INFORMATION_SCHEMA系统库锁相关数据表
‐‐ 查看事务 select * from INFORMATION_SCHEMA.INNODB_TRX; ‐‐ 查看锁 select * from INFORMATION_SCHEMA.INNODB_LOCKS; ‐‐ 查看锁等待 select * from INFORMATION_SCHEMA.INNODB_LOCK_WAITS; ‐‐ 释放锁,trx_mysql_thread_id可以从INNODB_TRX表里查看到 kill trx_mysql_thread_id ‐‐ 查看锁等待详细信息 show engine innodb status\G;
大多数情况mysql可以自动检测死锁并回滚产生死锁的那个事务,但是有些情况mysql没法自动检测死锁