• select...for update到底是加了行锁,还是表锁?


    前言

    前几天,知识星球中的一个小伙伴,问了我一个问题:在MySQL中,事务A中使用select...for update where id=1锁住了,某一条数据,事务还没提交,此时,事务B中去用select ... where id=1查询那条数据,会阻塞等待吗?

    select...for update在MySQL中,是一种悲观锁的用法,一般情况下,会锁住一行数据,但如果没有使用正确的话,也会把整张表锁住。

    其实,我之前也在实际项目中试过用,比如:积分兑换礼品的功能。

    今天跟大家一起聊聊select...for update这个话题,希望对你会有所帮助。

    1. 要什么要用行锁?

    假如现在有这样一种业务场景:用户A给你转账了2000元,用户B给你转账了3000元,而你的账户初始化金额是1000元。

    在事务1中会执行下面这条sql:

    update account set money=money+2000 
    where id=123;
    

    在事务2中执行下面这条sql:

    update account set money=money+3000 
    where id=123;
    

    这两条sql执行成功之后,你的money可能是:3000、4000、6000,这三种情况中的一种。

    你之前的想法是,用户A和用户B总共给你转账5000,最终你账户的钱应该是6000才对,3000和4000是怎么来的?

    假如事务1在执行update语句的过程中,事务2同时也在执行update语句。

    事务1中查询到money是1000,此外事务2也查询到money是1000。

    如果事务1先执行update语句,事务2后执行update语句,第一次update的3000,会被后面的4000覆盖掉,最终结果为4000。

    如果事务2先执行update语句,事务1后执行update语句,第一次update的4000,会被后面的3000覆盖掉,最终结果为3000。

    这两种情况都产生了严重的数据问题。

    我们需要有某种机制,保证事务1和事务2要顺序执行,不要一起执行。

    这就需要加锁了。

    目前MySQL中使用比较多的有:表锁、行锁和间隙锁。

    我们这个业务场景,非常时候使用行锁

    在事务1执行update语句的过程中,先要把某一行数据锁住,此时,其他的事务必须等待事务1执行完,提交了事务,才能获取那一行的数据。

    在MySQL中是通过select...for update语句来实现的行锁的功能。

    但如果你在实际工作中使用不正确,也容易把整张表锁住,严重影响性能。

    select...where...for update语句的用法是否正确,跟where条件中的参数有很大的关系。

    我们一起看看下面几种情况。

    假如user表现在有这样的数据库,数据库的版本是:8.0.21。

    创建的索引如下:

    其中id是主键字段,code是唯一索引字段,name是普通索引字段,其他的都是普通字段。

    2. 主键

    当where条件用的数据库主键时。

    例如开启一个事务1,在事务中更新id=1的用户的年龄:

    begin;
    select * from user where id=1 for update;
    update user set age=22 where id=1;
    

    where条件中的id是数据库的主键,并且使用for update关键字,加了一个行锁,这个事务没有commit。

    此时,开启了另外一个事务2,也更新id=1的用户的年龄:

    begin;
    update user set age=23 where id=1;
    commit;
    

    在执行事务2的sql语句的过程中,会一直等待事务1释放锁。

    如果事务1一直都不释放行锁,事务2最后会报下面这个异常:

    如果此时开始一个事务3,更新id=2的用户的年龄:

    begin;
    update user set age=23 where id=2;
    commit;
    

    执行结果如下:

    由于事务3中更新的另外一行数据,因此可以执行成功。

    说明使用for update关键字,锁住了主键id=1的那一行数据,对其他行的数据并没有影响。

    3. 唯一索引

    当where条件用的数据库唯一索引时。

    开启一个事务1,在事务中更新code=101的用户的年龄:

    begin;
    select * from user where code='101' for update;
    update user set age=22 where code='101';
    

    where条件中的code是数据库的唯一索引,并且使用for update关键字,加了一个行锁,这个事务没有commit。

    此时,开启了另外一个事务2,也更新code=101的用户的年龄:

    begin;
    update user set age=23 where code='101';
    commit;
    

    执行结果跟主键的情况是一样的。

    4. 普通索引

    当where条件用的数据库普通索引时。

    开启一个事务1,在事务中更新name=周星驰的用户的年龄:

    begin;
    select * from user where name='周星驰' for update;
    update user set age=22 where name='周星驰';
    

    where条件中的name是数据库的普通索引,并且使用for update关键字,加了一个行锁,这个事务没有commit。

    此时,开启了另外一个事务2,也更新name=周星驰的用户的年龄:

    begin;
    update user set age=23 where name='周星驰';
    commit;
    

    执行结果跟主键的情况也是一样的。

    5. 主键范围

    当where条件用的数据库主键范围时。

    开启一个事务1,在事务中更新id in (1,2)的用户的年龄:

    begin;
    select * from user where id in (1,2) for update;
    update user set age=22 where id in (1,2);
    

    where条件中的id是数据库的主键范围,并且使用for update关键字,加了多个行锁,这个事务没有commit。

    此时,开启了另外一个事务2,也更新id=1的用户的年龄:

    begin;
    update user set age=23 where id=1;
    commit;
    

    执行结果跟主键的情况也是一样的。

    此时,开启了另外一个事务2,也更新id=2的用户的年龄:

    begin;
    update user set age=23 where id=2;
    commit;
    

    执行结果跟主键的情况也是一样的。

    6. 普通字段

    当where条件用的数据库普通字段时。

    该字段既不是主键,也不是索引。

    开启一个事务1,在事务中更新age=22的用户的年龄:

    begin;
    select * from user where age=22 for update;
    update user set age=22 where age=22 ;
    

    where条件中的age是数据库的普通字段,并且使用for update关键字,加的是表锁,这个事务没有commit。

    此时,开启了另外一个事务2,也更新age=22的用户的年龄:

    begin;
    update user set age=23 where age=22 ;
    commit;
    

    此时,执行事务2时,会一直阻塞等待事务1释放锁。

    调整一下sql条件,查询条件改成age=23:

    begin;
    update user set age=23 where age=23 ;
    commit;
    

    此时,行事务3时,也会一直阻塞等待事务1释放锁。

    也就是说,在for update语句中,使用普通字段作为查询条件时,加的是表锁,而并非行锁。

    7. 空数据

    当where条件查询的数据不存在时,会发生什么呢?

    开启一个事务1,在事务中更新id=66的用户的年龄:

    begin;
    select * from user where id=66 for update;
    update user set age=22 where id=66 ;
    

    这条数据是不存在的。

    此时,开启了另外一个事务2,也更新id=66的用户的年龄:

    begin;
    update user set age=23 where id=66 ;
    commit;
    

    执行结果:

    执行成功了,说明这种情况没有加锁。

    总结

    最后给大家总结一下select...for update加锁的情况:

    1. 主键字段:加行锁。
    2. 唯一索引字段:加行锁。
    3. 普通索引字段:加行锁。
    4. 主键范围:加多个行锁。
    5. 普通字段:加表锁。
    6. 查询空数据:不加锁。

    如果事务1加了行锁,一直没有释放锁,事务2操作相同行的数据时,会一直等待直到超时。

    如果事务1加了表锁,一直没有释放锁,事务2不管操作的是哪一行数据,都会一直等待直到超时。

    最后说一句(求关注,别白嫖我)

    如果这篇文章对您有所帮助,或者有所启发的话,帮忙扫描下发二维码关注一下,您的支持是我坚持写作最大的动力。

    求一键三连:点赞、转发、在看。

    关注公众号:【苏三说技术】,在公众号中回复:面试、代码神器、开发手册、时间管理有超赞的粉丝福利,另外回复:加群,可以跟很多BAT大厂的前辈交流和学习。

    最后欢迎大家加入苏三的知识星球【Java突击队】,一起学习。

    星球中有很多独家的干货内容,比如:Java后端学习路线,分享实战项目,源码分析,百万级系统设计,系统上线的一些坑,MQ专题,真实面试题,每天都会回答大家提出的问题,免费修改简历,免费回答工作中的问题。

    星球目前开通了9个优质专栏:技术选型、系统设计、踩坑分享、工作实战、底层原理、Spring源码解读、痛点问题、高频面试题 和 性能优化。

  • 相关阅读:
    stm32学习笔记:EXIT中断
    Jmeter进阶使用指南-使用参数化
    JVM 发生 OOM 的 8 种原因、及解决办法
    ubuntu1804配置OAK-D相机的ROS以及depthai_ros环境
    分布式锁选型+缓存db一致性
    Java核心编程(17)
    SpringBoot实用开发篇复习3
    读《追问》
    javaweb教师人事管理系统的设计
    表白墙(web版)
  • 原文地址:https://www.cnblogs.com/12lisu/p/17787892.html