行锁和表锁的区别

行锁


在MySQL的InnoDB引擎支持行锁,与Oracle不同,MySQL的行锁是通过索引加载的,也就是说,行锁是加在索引响应的行上的,要是对应的SQL语句没有走索引,则会全表扫描,行锁则无法实现,取而代之的是表锁,此时其它事务无法对当前表进行更新或插入操作。


特征:


锁冲突概率低,并发性高,但是会有死锁的情况出现。


行锁的实现需要注意:


  1. 行锁必须有索引才能实现,否则会自动锁全表,那么就不是行锁了。
  2. 两个事务不能锁同一个索引。
  3. insert,delete,update在事务中都会自动默认加上排它锁。


行锁场景:


A用户消费,service层先查询该用户的账户余额,若余额足够,则进行后续的扣款操作;这种情况查询的时候应该对该记录进行加锁。


否则,B用户在A用户查询后消费前先一步将A用户账号上的钱转走,而此时A用户已经进行了用户余额是否足够的判断,则可能会出现余额已经不足但却扣款成功的情况。


为了避免此情况,需要在A用户操作该记录的时候进行for update加锁


for update


如果在一条select语句后加上for update,则查询到的数据会被加上一条排它锁,其它事务可以读取,但不能进行更新和插入操作


表锁


顾名思义,表锁就是一锁锁一整张表,在表被锁定期间,其他事务不能对该表进行操作,必须等当前表的锁被释放后才能进行操作。表锁响应的是非索引字段,即全表扫描,全表扫描时锁定整张表,sql语句可以通过执行计划看出扫描了多少条记录。


特征:


不会出现死锁,发生锁冲突几率高,并发低。


MyISAM引擎


MyISAM在执行查询语句(select)前,会自动给涉及的所有表加读锁,在执行增删改操作前,会自动给涉及的表加写锁。


MySQL的表级锁有两种模式:


  • 表共享读锁
  • 表独占写锁


读锁会阻塞写,写锁会阻塞读和写


  • 对MyISAM表的读操作,不会阻塞其它进程对同一表的读请求,但会阻塞对同一表的写请求。只有当读锁释放后,才会执行其它进程的写操作。
  • 对MyISAM表的写操作,会阻塞其它进程对同一表的读和写操作,只有当写锁释放后,才会执行其它进程的读写操作。


MyISAM不适合做写为主表的引擎,因为写锁后,其它线程不能做任何操作,大量的更新会使查询很难得到锁,从而造成永远阻塞


如何加锁?


MyISAM在执行查询语句(SELECT)前,会自动给涉及的所有表加读锁,在执行更新操作(UPDATE、DELETE、INSERT等)前,会自动给涉及的表加写锁,这个过程并不需要用户干预,因此用户一般不需要直接用LOCK TABLE命令给MyISAM表显式加锁。


显式加锁:


上共享锁(读锁)的写法:lock in share mode,例如:


select  math from zje where math>60 lock in share mode;


上排它锁(写锁)的写法:for update,例如:


select math from zje where math >60 for update;




举报
评论 0