update语句未使用索引的后果

update语句未使用索引的后果

网上有一种说法:mysql的update语句不使用索引,会锁表;使用索引,会锁行。这种说法是非常不准确的。今天来简单分析一下。

本文使用的mysql版本为8.0.32,innodb存储引擎。

例1

本例中的表结构如下:

sql 复制代码
CREATE TABLE `t7` (
  `id` int NOT NULL AUTO_INCREMENT,
  `a` int DEFAULT NULL,
  `b` int DEFAULT NULL,
  `c` int DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `b` (`b`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci

当前数据库中的记录为:

在事务中更新记录:

sql 复制代码
begin;
update t7 set c=11 where a=2 and c=4;

下面,我们来针对这个事务涉及到的锁进行分析。

update语句是否涉及表级锁

首先,update语句执行过程中会涉及到表级锁吗? 答案是会,update语句在执行过程中,会涉及到两种表级锁: 元数据锁和表级意向锁。

MDL锁/元数据锁

MDL锁的主要作用是保证表元数据的一致性。

MySQL server会持有元数据锁,直到事务结束。如果另一个会话尝试对相关表进行DDL操作或者LOCK TABLE,将会阻塞。

关于MDL的详细介绍,可以查看官方文档:元数据锁

我们可以通过查询performance_schema.metadata_locks表来查看元数据锁信息:

sql 复制代码
select * from performance_schema.metadata_locks\G;

查询结果:

表级意向锁

表级意向锁(IS,IX),是为了在以后加表级别的S锁和X锁时可以快速判断表中记录是否被上锁而提出的。 IS与IS、IX与IX、IS与IX之间,相互兼容。 可以通过查询performance_schema.data_locks来获取表及意向锁信息。

sql 复制代码
select * from performance_schema.data_locks\G;

查询结果:

需要注意的是,无论是MDL锁,还是表级意向锁,都不会影响update语句的执行,也就是说,update语句不会因为这两种锁而阻塞。

update语句没有使用到索引究竟会发生什么

update语句如果没有使用到索引,会进行全表扫描,会对表中/聚簇索引的每一行记录进行加锁,这会耗费较长时间。如果用show engine innodb status查看,你可能会发现语句长时间处于searching rows for update状态。 更严重的是,这会带来严重的并发问题。

  • 如果是用RC级别,如果被读取的记录不满足where条件,对应的记录锁会被立即释放。
  • 如果是用RR级别,即使被读取的记录不满足where条件,对应的记录锁也会不会释放,直到事务结束。同时,这个事务也会持有gap锁,阻塞其他事务对这个范围的插入操作。

如果是用RR级别,并发能力会大大降低。这也是很多公司都不会使用默认的RR级别,而改为使用RC级别的原因。

使用到了索引,就一定没问题吗

即使用到了索引,如果索引区分度不高,并发时依然有可能造成较多的锁等待。 此外,如果使用到了索引合并(index merge),还可能会增加死锁的概率。

所以,对于update语句,一定要尽量使用索引,而且要保证索引区分度较高,最好是保证where条件中的每一列都在索引中。

注:此文原载于本人个人网站链接地址

参考资料:

  1. MySQL官方手册-元数据锁

本文由mdnice多平台发布

相关推荐
煎饼小狗10 分钟前
Redis五大基本类型——Zset有序集合命令详解(命令用法详解+思维导图详解)
数据库·redis·缓存
永乐春秋26 分钟前
WEB-通用漏洞&SQL注入&CTF&二次&堆叠&DNS带外
数据库·sql
打鱼又晒网1 小时前
【MySQL】数据库精细化讲解:内置函数知识穿透与深度学习解析
数据库·mysql
大白要努力!1 小时前
android 使用SQLiteOpenHelper 如何优化数据库的性能
android·数据库·oracle
tatasix2 小时前
MySQL UPDATE语句执行链路解析
数据库·mysql
南城花随雪。2 小时前
硬盘(HDD)与固态硬盘(SSD)详细解读
数据库
儿时可乖了2 小时前
使用 Java 操作 SQLite 数据库
java·数据库·sqlite
懒是一种态度2 小时前
Golang 调用 mongodb 的函数
数据库·mongodb·golang
天海华兮2 小时前
mysql 去重 补全 取出重复 变量 函数 和存储过程
数据库·mysql
gma9993 小时前
Etcd 框架
数据库·etcd