mysql锁机制分为表级锁和行级锁,本文就和你们分享一下我对mysql中行级锁中的共享锁与排他锁进行分享交流。mysql
共享锁又称为读锁,简称S锁,顾名思义,共享锁就是多个事务对于同一数据能够共享一把锁,都能访问到数据,可是只能读不能修改。sql
排他锁又称为写锁,简称X锁,顾名思义,排他锁就是不能与其余所并存,如一个事务获取了一个数据行的排他锁,其余事务就不能再获取该行的其余锁,包括共享锁和排他锁,可是获取排他锁的事务是能够对数据就行读取和修改。测试
对于共享锁你们可能很好理解,就是多个事务只能读数据不能改数据,对于排他锁你们的理解可能就有些差异,我当初就犯了一个错误,觉得排他锁锁住一行数据后,其余事务就不能读取和修改该行数据,其实不是这样的。排他锁指的是一个事务在一行数据加上排他锁后,其余事务不能再在其上加其余的锁。mysql InnoDB引擎默认的修改数据语句,update,delete,insert都会自动给涉及到的数据加上排他锁,select语句默认不会加任何锁类型,若是加排他锁可使用select ...for update语句,加共享锁可使用select ... lock in share mode语句。因此加过排他锁的数据行在其余事务种是不能修改数据的,也不能经过for update和lock in share mode锁的方式查询数据,但能够直接经过select ...from...查询数据,由于普通查询没有任何锁机制。3d
说了这么多,我们来看下如下简单的例子:blog
咱们有以下测试数据事务
如今咱们对id=1的数据行排他查询,这里会使用begin开启事务,而不会看见我关闭事务,这样作是用来测试,由于提交事务或回滚事务就会释放锁。bfc
打开一个查询窗口date
会查询到一条数据,如今打开另外一个查询窗口,对同一数据分别使用排他查和共享锁查询两种方式查询select
排他查im
共享查
咱们看到开了排他锁查询和共享锁查询都会处于阻塞状态,由于id=1的数据已经被加上了排他锁,此处阻塞是等待排他锁释放。
若是咱们直接使用如下查询呢
咱们看到是能够查询到数据的。
咱们再看一下一个事务获取了共享锁,在其余查询中也只能加共享锁或不加锁。
咱们看到是能够查询数据的,但加排他锁就查不到,由于排他锁与共享锁不能存在同一数据上。
最后咱们验证下上面说的mysql InnoDb引擎中update,delete,insert语句自动加排他锁的问题,
此时共享查询处于阻塞,等待排它锁的释放,可是用普通查询能查到数据,由于没用上锁机制不与排他锁互斥,但查到的数据是修改数据以前的老数据。
而后咱们提交数据,释放排他锁看下修改后的数据,此时可用排他查,共享查和普通查询, 由于事务提交后该行数据释放排他锁,下面就只显示普通查询,其余的同窗们本身去验证。
能够看到结果与预期的同样。
以上为我对mysql中共享锁与排他锁的我的理解,有不正确的地方还但愿各位指正。