MySQL

MySQL共享锁、排他锁、悲观锁、乐观锁及其使用场景

一、相关名词

表级锁(锁定整个表)
页级锁(锁定一页)
行级锁(锁定一行)
共享锁(S锁,MyISAM 叫做读锁)
排他锁(X锁,MyISAM 叫做写锁)
悲观锁(抽象性,不真实存在这个锁)
乐观锁(抽象性,不真实存在这个锁)

二、InnoDB与MyISAM

Mysql 在5.5之前默认使用 MyISAM 存储引擎,之后使用 InnoDB 。查看当前存储引擎:

show variables like '%storage_engine%';

MyISAM 操作数据都是使用的表锁,你更新一条记录就要锁整个表,导致性能较低,并发不高。当然同时它也不会存在死锁问题。

而 InnoDB 与 MyISAM 的最大不同有两点:一是 InnoDB支持事务;二是 InnoDB 采用了行级锁。也就是你需要修改哪行,就可以只锁定哪行。

在 Mysql 中,行级锁并不是直接锁记录,而是锁索引。索引分为主键索引和非主键索引两种,如果一条sql 语句操作了主键索引,Mysql 就会锁定这条主键索引;如果一条语句操作了非主键索引,MySQL会先锁定该非主键索引,再锁定相关的主键索引。

InnoDB 行锁是通过给索引项加锁实现的,如果没有索引,InnoDB 会通过隐藏的聚簇索引来对记录加锁。也就是说:如果不通过索引条件检索数据,那么InnoDB将对表中所有数据加锁,实际效果跟表锁一样。因为没有了索引,找到某一条记录就得扫描全表,要扫描全表,就得锁定表。

三、共享锁与排他锁

1.首先说明:数据库的增删改操作默认都会加排他锁,而查询不会加任何锁

共享锁:对某一资源加共享锁,自身可以读该资源,其他人也可以读该资源(也可以再继续加共享锁,即 共享锁可多个共存),但无法修改。要想修改就必须等所有共享锁都释放完之后。语法为:
select * from table lock in share mode
排他锁:对某一资源加排他锁,自身可以进行增删改查,其他人无法进行任何操作。语法为:
select * from table for update -- 增删改自动加了排他锁

2.下面援引例子说明(援自:http://blog.csdn.net/samjustin1/article/details/52210125):

这里用T1代表一个数据库执行请求,T2代表另一个请求,也可以理解为T1为一个线程,T2 为另一个线程。

例1:

T1: select * from table lock in share mode
(假设查询会花很长时间,下面的例子也都这么假设)

T2: update table set column1='hello'

过程:

1 条评论

  1. 头像

    好高端的东西 ,不懂

评论已关闭。

相关推荐