社区所有版块导航
Python
python开源   Django   Python   DjangoApp   pycharm  
DATA
docker   Elasticsearch  
aigc
aigc   chatgpt  
WEB开发
linux   MongoDB   Redis   DATABASE   NGINX   其他Web框架   web工具   zookeeper   tornado   NoSql   Bootstrap   js   peewee   Git   bottle   IE   MQ   Jquery  
机器学习
机器学习算法  
Python88.com
反馈   公告   社区推广  
产品
短视频  
印度
印度  
Py学习  »  DATABASE

MYSQL中select for update是锁表还是锁行

码小辫 • 1 年前 • 260 次点击  
   

大家好,我是小码同学,一个大龄程序员, 在这里我将分享一些知识干货以及互联网一些热点话题,感兴趣的话就关注我吧,希望对你有所帮助。


把 码小辫 设为“星标”咱们就自己人了!有酒一起喝!有肉一起吃!

在并发一致性控制场景中,我们常常用for update悲观锁来进行一致性的保证,但是如果不了解它的机制,就进行使用,很容易出现事故,比如for update进行了锁表导致其他请求只能等待,从而拖垮系统,因此了解它的原理是非常必要的,下面我们通过一系列示例进行测试,来看看到底是什么场景下锁表什么场景下锁行

验证

示例说明

创建一个账户表,插入基础数据,以唯一索引普通索引主键普通字段4 个维度进行select ... for update查询,查看是进行锁表还是锁行

表创建

创建一个账户表,指定account_no为唯一索引、id为主键、user_no 为普通字段、curreny为普通索引

CREATE TABLE `account_info` (
  `id` int NOT NULL AUTO_INCREMENT COMMENT 'ID' ,
 `account_no` int NOT NULL COMMENT '账户编号',
 `user_no` varchar(32NOT NULL COMMENT '用户 Id',
 `currency` varchar(10NOT NULL COMMENT '币种',
  `amount` DECIMAL(10,2NOT NULL COMMENT '金额',
 `freeze_amount` DECIMAL(10,2NOT NULL COMMENT '冻结金额',
  `create_time` datetime(6NOT NULL DEFAULT CURRENT_TIMESTAMP(6COMMENT '创建时间',
  `update_time` datetime(6NOT NULL DEFAULT CURRENT_TIMESTAMP(6ON UPDATE CURRENT_TIMESTAMP(6COMMENT '修改时间',
  PRIMARY KEY (`id`USING BTREE,
 UNIQUE KEY `uni_idx_account_no` (`account_no`) ,
 KEY `idx_currency_` (`currency`
ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='账户信息表';

插入基础数据

insert into account_info values (1,1,'ur001','RMB',100,0,now(),now());
insert into account_info values (2,2,'ur002','RMB',1000,0,now(),now());
insert into account_info values (3,3,'ur002','DOLLAR',200,0,now(),now());

根据主键查询

在事务 1 中,根据主键id=1 进行 for update查询时,事务2、事务 3 都进行阻塞,而事务 4 由于更新的id=2 所以成功,因此判定,根据主键进行 for update 查询时是行锁

根据唯一索引查询

在事务 1 中,根据唯一索引account_no=1 进行 for update查询时,事务2、事务 3 都进行阻塞,而事务 4 由于更新的account_no=2 所以成功,因此判定,根据唯一索引进行 for update 查询时是行锁

根据普通索引查询

在事务 1 中,根据普通索引currency='RMB' 进行 for update查询时,事务2、事务 3 都进行阻塞,而事务 4 由于更新的currency='DOLLAR`所以成功,因此判定,根据普通索引进行 for update 查询时是行锁

根据普通字段查询

在事务 1 中,根据普通字段user_no='ur001' 进行 for update查询时,事务2、事务 3 都进行阻塞,而事务 4查询的是user_no='ur002'也进行阻塞,因此判定,根据普通字段进行 for update 查询时是表锁

总结

如果查询条件是索引/主键字段,那么select ..... for update会进行行锁

如果查询条件是普通字段(没有索引/主键),那么select ..... for update会进行锁表,这点一定要注意。


往期推荐

4 种 MySQL 同步 ES 方案,yyds!

我承认,李一舟的课我买了!

第一次使用缓存,因为没预热,翻车了

马斯克抱怨新买的笔记本强制登录微软账号还无法跳过

为什么计算机需要十六进制?

这里有最新前沿技术资讯、技术干货等内容

点这里 ↓↓↓ 记得 关注✔ 标星⭐ 哦


Python社区是高质量的Python/Django开发社区
本文地址:http://www.python88.com/topic/167558
 
260 次点击