可能遇到这样的场景:数据库某一条记录的数据是3, 两个线程同时更新这条数据:查询出数据的值3,将它减1,得到2,再写回数据库。两个线程更新完后最后数据库数据可能是2。解决这个问题就要用到数据库自带的锁机制for update或者乐观锁。
1、for update
mysql 锁定记录行的方式for update是一个悲观锁而且是排它锁。for update锁定的行,在当前事务没有提交之前,其他事务即不能更新锁定的行,会被阻塞等待当前事务释放锁。 举例:
第一步:打开一个事务,关闭自动提交后,使用for update锁定一条记录
select * from student where id = 1 for update;
第二步:再打开一个窗口,再次执行相同的记录,发现会被阻塞等待锁。也就是说这个for update是个排他锁。
select * from student where id = 1 for update;
第三步:再打开一个新窗口,更新第一步被for update查询的记录,也被阻塞。
update student set age=12 where id =1;
总结:select * from xxx for update悲观锁某条记录一旦被锁定,其他事务尝试用select * from xxx for update再次获取记录上的锁会被阻塞,直到锁被持有的事务释放或者超时退出。其他事务尝试更新这条记录也会被阻塞等待锁释放直到超时退出为止。
第四步:让持有id=1记录锁的事务更新这条记录的age=12,但是不提交事务。
update student set age=12 where id =1;
select * from student where id = 1;
第五步:不持有锁的事务查询第四步持有锁更新记录,但是没有提交的记录。
简单查询,不使用for update
select * from student where id = 1;
总结2:for update锁定的记录,对于没有持有锁的事务来说,如果不使用for update竞争锁的话可以正常查询。根据mysql默认事务隔离级别:可重复读,读取的是未提交的数据。
|