什么是“锁”已经锁的作用?
“锁”顾名思义就是锁定的意思,在事务ACID过程中,“锁”和“隔离级别”一起来实现“I”隔离性的作用。


锁的粒度:

  1. MyIasm:低并发锁——表级锁
  2. 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

注:本文有部分来自网络摘抄。

最后修改于 2019-08-09 14:14:51
如果觉得我的文章对你有用,请随意赞赏
扫一扫支付
上一篇