MySQL Execution Plan--IN子查询对UPDATE语句影响
问题描述
在系统中发现一条执行时间为为44652.060734秒(12.5小时)的慢SQL,SQL语句为:
UPDATE
ob_internal_task
SET
OPERATE_STATUS = 0
WHERE business_no IN
(
SELECT
outbound_no
FROM
ob_internal_orderstatus
WHERE wave_no = 'xxxxx'
AND VSTATE = 10
AND outbound_no <> 'xxxxxxxx'
)
AND business_type = 20;
对于执行计划为:
*************************** 1. row ***************************
id: 1
select_type: UPDATE
table: ob_internal_task
partitions: NULL
type: index
possible_keys: NULL
key: PRIMARY
key_len: 8
ref: NULL
rows: 1657847
filtered: 100.00
Extra: Using where
*************************** 2. row ***************************
id: 2
select_type: DEPENDENT SUBQUERY
table: ob_internal_orderstatus
partitions: NULL
type: ALL
possible_keys: NULL
key: NULL
key_len: NULL
ref: NULL
rows: 622710
filtered: 0.09
Extra: Using where
由于两个表上都使用全表扫描,需要循环外表ob_internal_task记录1654884次*内表ob_internal_orderstatus记录622596次=1030324158864(预估影响行数)
解决步骤
表ob_internal_task列business_no上有索引,没有被正确使用,而表上ob_internal_orderstatus缺少合适索引,因此首先优化IN子查询内部,创建索引:
ALTER TABLE ob_internal_orderstatus
ADD INDEX `IDX_wave_no_VSTATE1` (`WAVE_NO`,`VSTATE`,`OUTBOUND_NO`);
创建完索引后执行计划变更为:
*************************** 1. row ***************************
id: 1
select_type: UPDATE
table: ob_internal_task
partitions: NULL
type: index
possible_keys: NULL
key: PRIMARY
key_len: 8
ref: NULL
rows: 1656801
filtered: 100.00
Extra: Using where
*************************** 2. row ***************************
id: 2
select_type: DEPENDENT SUBQUERY
table: ob_internal_orderstatus
partitions: NULL
type: ref
possible_keys: IDX_wave_no_VSTATE1
key: IDX_wave_no_VSTATE1
key_len: 131
ref: const,const,func
rows: 1
filtered: 100.00
Extra: Using where; Using index
外表仍然使用全表扫描,检查两表关联列,发现都是VARCHAR类型,不存在隐式转换。
将UPDATE语句调整为SELECT测试,调整后的SQL为:
SELECT
*
FROM
ob_internal_task
WHERE business_no IN
(
SELECT
outbound_no
FROM
ob_internal_orderstatus
WHERE wave_no = 'BC20190523155122'
AND VSTATE = 10
AND outbound_no <> ''
)
AND business_type = 20;
对于执行计划为:
*************************** 1. ROW ***************************
id: 1
select_type: SIMPLE
TABLE: ob_internal_orderstatus
PARTITIONS: NULL
TYPE: ref
possible_keys: IDX_wave_no_VSTATE1
KEY: IDX_wave_no_VSTATE1
key_len: 68
ref: const,const
ROWS: 26
filtered: 90.00
Extra: USING WHERE; USING INDEX; LooseScan
*************************** 2. ROW ***************************
id: 1
select_type: SIMPLE
TABLE: ob_internal_task
PARTITIONS: NULL
TYPE: ref
possible_keys: idx_business_type,idx_business_no
KEY: idx_business_no
key_len: 303
ref: innerdelivery.ob_internal_orderstatus.OUTBOUND_NO
ROWS: 3
filtered: 10.00
Extra: USING INDEX CONDITION; USING WHERE
对比UPDATE和SELECT语句的执行计划,发现内外表的位置发生变化,同时也导致表ob_internal_task上索引idx_business_no能被正常使用。
考虑到IN语句的影响,尝试将UPDATE语句中的依赖子查询调整为关联查询,调整后SQL为:
UPDATE ob_internal_task T1
INNER JOIN
(
SELECT
outbound_no
FROM
ob_internal_orderstatus
WHERE wave_no = 'BC20190523155122'
AND VSTATE = 10
AND outbound_no <> ''
) AS T2
ON T1.business_no = T2.outbound_no
SET T1.OPERATE_STATUS = 0
WHERE T1.business_type = 20;
调整后的执行计划为:
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: ob_internal_orderstatus
partitions: NULL
type: range
possible_keys: IDX_wave_no_VSTATE1
key: IDX_wave_no_VSTATE1
key_len: 131
ref: NULL
rows: 25
filtered: 100.00
Extra: Using where; Using index
*************************** 2. row ***************************
id: 1
select_type: UPDATE
table: T1
partitions: NULL
type: ref
possible_keys: idx_business_type,idx_business_no
key: idx_business_no
key_len: 303
ref: func
rows: 3
filtered: 10.00
Extra: Using where
使用关联更新的UPDATE语句能够正常使用索引,执行时间由原来44652.060734秒(12.5小时)缩短至的在30ms内。
PS1:相关表的统计信息未存在明显异常,排除统计信息导致执行计划异常。
优化总结
1、在优化DELETE/UPDATE语句时,通常会将DELETE/UPDATE语句改写成SELECT语句进行测试,以避免测试操作导致数据变更,需要注意两者的执行计划可能不同。
2、当IN语句生产的执行计划为DEPENDENT SUBQUERY类型时,需要考虑外表的循环数量,如果外表循环次数较大,可以考虑调整SQL语句(如关联查询)来优化内外表位置。
MySQL Execution Plan--IN子查询对UPDATE语句影响的更多相关文章
- 子查询在UPDATE 语句中的应用
在UPDATE语句中可以在更新列表中以及WHERE语句使用子查询.下面演示一个将图书的出版日期全部更新为所有图书中的最新出版日期,SQL语句如下: UPDATE T_Book SET FYearPub ...
- mysql sql_safe_updates 不支持子查询的更新。
考虑到开发人员有时候不小心误更新数据,要求线上库的 MySQL 实例都设置 sql_safe_updates=1 来避免没有索引的 update.delete. 结果有一天开发发现下面的一个SQL 没 ...
- mssql sql高效关联子查询的update 批量更新
/* 使用带关联子查询的Update更新 --1.创建测试表 create TABLE Table1 ( a varchar(10), b varchar(10), ...
- MySQL(八)子查询和分组查询
一.子查询 1.子查询(subquery):嵌套在其他查询中的查询. 例如:select user_id from usertable where mobile_no in (select mobil ...
- sql子查询 嵌套SELECT语句
嵌套SELECT语句也叫子查询,一个 SELECT 语句的查询结果能够作为另一个语句的输入值.子查询不但能够出现在Where子句中,也能够出现在from子句中,作为一个临时表使用,也能够出现在sele ...
- MySQL锁类型以及子查询锁表问题、解锁
MySQL中select * for update锁表的范围 MySQL中select * for update锁表的问题 由于InnoDB预设是Row-Level Lock,所以只有「明确」的指定主 ...
- Mysql查询优化器之关于子查询的优化
下面这些sql都含有子查询: mysql> select * from t1 where a in (select a from t2); mysql> select * from (se ...
- Mysql - 性能优化之子查询
记得在做项目的时候, 听到过一句话, 尽量不要使用子查询, 那么这一篇就来看一下, 这句话是否是正确的. 那在这之前, 需要介绍一些概念性东西和mysql对语句的大致处理. 当Mysql Server ...
- MySQL 使用JOIN优化子查询
1.数据准备 mysql> select * from student; +----+--------+----------+---------+-------------+ | id | na ...
随机推荐
- java多线程(六)线程控制类
1. 多线程控制类 为了保证多线程的三个特性,Java引入了很多线程控制机制,下面介绍其中常用的几种: l ThreadLocal l 原子类 l Lock类 l Volatile关键字 ...
- fatal error: sys/videoio.h: No such file or directory
Determining if the include file sys/videoio.h exists failed with the following output:Change Dir: /h ...
- 为什么java里面经常作List判断的时候,既要判断list不为null,又要判断size>0呢?
没有考虑到具体的问题上面,我们单纯的来讲: 为什么java里面经常作List判断的时候,既要判断list不为null,又要判断size>0呢? list == null 说明list没有初始化( ...
- C# 将DataTable数据写入到txt文件中
见代码: /// <summary> /// 将DataTable里面的内容写入txt文件 /// </summary> /// <param name="dt ...
- 设计模式php+java版本(1) 基础篇 七大原则
2019年9月6日11:15:46 关于设计模式,其实就是编程思想的一个体现,有比较完善的编程思想写出的项目代码和没有编程思想的写出的差距巨大,代码的可读性,可维护性,可扩展性天差地别,有些刚接触的编 ...
- Python3使用random生成随机数
本文介绍使用Python3中的random库生成随机数.随机小数.随机序列.随机字符串以及扑克洗牌等方法. 一.生成随机浮点数或小数 1.#生成0-1之间的浮点数 import random rnd ...
- ajax处理csrf的三种方式
方式一: $.post({ url: '/get_result/', data: { value0: $('#v1').val(), value1: $('#v2').val(), csrfmiddl ...
- 移动端实现裁剪图片生成base64图片(可缩放)
移动端实现裁剪图片生成base64图片(可缩放)<pre><!DOCTYPE html><html lang="en"> <head> ...
- python 异常处理(25)
在python开发中,代码书写时难免有疏忽或者意向不到的bug,导致程序run的过程中有可能会直接崩溃:然后对于程序猿而言,程序因bug崩溃是家常便饭,为了增加程序的健壮性,防止程序崩溃,我们可以对程 ...
- Docker下安装zookeeper(单机 & 集群)
启动Docker后,先看一下我们有哪些选择. 有官方的当然选择官方啦~ 下载: [root@localhost admin]# docker pull zookeeper Using default ...