什么是“锁”已经锁的作用?
“锁”顾名思义就是锁定的意思,在事务ACID过程中,“锁”和“隔离级别”一起来实现“I”隔离性的作用。
锁的粒度:
- MyIasm:低并发锁——表级锁
- Innodb:高并发锁——行级锁
InnoDB引擎的锁机制
之所以以InnoDB为主介绍锁,是因为InnoDB支持事务,支持行锁和表锁用的比较多,Myisam不支持事务,只支持表锁。
共享锁(S):允许一个事务去读一行,阻止其他事务获得相同数据集的排他锁。
排他锁(X):允许获得排他锁的事务更新数据,阻止其他事务取得相同数据集的共享读锁和排他写锁。
意向共享锁(IS):事务打算给数据行加行共享锁,事务在给一个数据行加共享锁前必须先取得该表的IS锁。
意向排他锁(IX):事务打算给数据行加行排他锁,事务在给一个数据行加排他锁前必须先取得该表的IX锁。
说明:
1)共享锁和排他锁都是行锁,意向锁都是表锁,应用中我们只会使用到共享锁和排他锁,意向锁是mysql内部使用的,不需要用户干预。
2)对于UPDATE、DELETE和INSERT语句,InnoDB会自动给涉及数据集加排他锁(X);对于普通SELECT语句,InnoDB不会加任何锁,事务可以通过以下语句显示给记录集加共享锁或排他锁。
共享锁(S):SELECT * FROM table_name WHERE ... LOCK IN SHARE MODE。
排他锁(X):SELECT * FROM table_name WHERE ... FOR UPDATE。
3)InnoDB行锁是通过给索引上的索引项加锁来实现的,因此InnoDB这种行锁实现特点意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁!。
行级锁的三种算法:
- Record Lock
Record Lock表示在单个记录行上加锁,Record Lock总是会锁住索引记录,如果InnoDB存储引擎表在建立的时候没有设置任何一个索引,那么这是InnoDB存储引擎会使用隐式的主键来进行锁定。 - Gap Lock
间隙锁,锁定一个范围,但不包含记录本身。 - Next-Key Lock
Next-Key Lock的锁定技术成为Next-Key Locking。其设置的目的是为了解决幻象问题。Next-Key Lock是结合了Gap Lock和Record Lock的一种锁定算法。
当查询的索引含有唯一属性的时候,InnoDB存储引擎会对Next-Key Lock进行优化,将其降为Record Lock,即仅仅锁住索引本身,而不是范围,从而提高并发效率。
Gap Lock的作用是为了阻止多个事务将记录插入到同一个范围内,这样会导致幻读的产生。用户可以通过以下两种服务显式地关闭Gap Lock。
1、 将事务的隔离级别设置为READ COMMITTED
2、 将参数innodb_locks_unsafe_for_binlog设置为1
除了外键约束和唯一性检查需要的Gap Lock,其余情况仅使用Record Lock进行锁定。这样做破坏了事务的隔离性,并且对于replication会导致主动数据的不一致。
对于唯一值的锁定,Next-Key Lock降级为Record Lock仅存在于查询所有的唯一索引列。若唯一索引由多个列组成,而查询仅是查找多个唯一索引列中的其中一个,那么查询其实是range类型,而不是point类型的查询。此时InnoDB存储引擎依然使用Next-Key Lock进行锁定。
事务隔离级别,解决什么并发问题,以及存在什么并发问题
Mysql默认的事务隔离级别为repeatable_read
mysql> show variables like '%tx_%';
+---------------+-----------------+
| Variable_name | Value |
+---------------+-----------------+
| tx_isolation | REPEATABLE-READ |
| tx_read_only | OFF |
+---------------+-----------------+
2 rows in set (0.01 sec)
(1)READ_UNCOMMITTED (RU 为提交读) 造成脏读
这是事务最低的隔离级别,它充许另外一个事务可以看到这个事务未提交的数据。
解决第一类丢失更新的问题,但是会出现脏读、不可重复读、第二类丢失更新的问题,幻读 。
(2)READ_COMMITTED (RC 提交读) 造成幻读
保证一个事务修改的数据提交后才能被另外一个事务读取,即另外一个事务不能读取该事务未提交的数据。
解决第一类丢失更新和脏读的问题,但会出现不可重复读、第二类丢失更新的问题,幻读问题
(3)REPEATABLE_READ (RR 可重复读)
保证一个事务相同条件下前后两次获取的数据是一致的
解决第一类丢失更新,脏读、不可重复读、第二类丢失更新的问题,但会出幻读。
(4)SERIALIZABLE (SERI 序列化 完全串行化执行对应得事务)
事务被处理为顺序执行。
解决所有问题
LU丢失更新 | DR脏读 | NRR非重复读 | SLU二类丢失更新 | PR幻象读 | |
未提交读 RU | Y | Y | Y | Y | Y |
提交读 RC | N | N | Y | Y | Y |
可重复读 RR | N | N | N | N | Y |
串行读 S | N | N | N | N | N |
注:本文有部分来自网络摘抄。

