问题描述

在系统中发现一条执行时间为为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语句影响的更多相关文章

  1. 子查询在UPDATE 语句中的应用

    在UPDATE语句中可以在更新列表中以及WHERE语句使用子查询.下面演示一个将图书的出版日期全部更新为所有图书中的最新出版日期,SQL语句如下: UPDATE T_Book SET FYearPub ...

  2. mysql sql_safe_updates 不支持子查询的更新。

    考虑到开发人员有时候不小心误更新数据,要求线上库的 MySQL 实例都设置 sql_safe_updates=1 来避免没有索引的 update.delete. 结果有一天开发发现下面的一个SQL 没 ...

  3. mssql sql高效关联子查询的update 批量更新

    /* 使用带关联子查询的Update更新     --1.创建测试表 create TABLE Table1     (     a varchar(10),     b varchar(10),   ...

  4. MySQL(八)子查询和分组查询

    一.子查询 1.子查询(subquery):嵌套在其他查询中的查询. 例如:select user_id from usertable where mobile_no in (select mobil ...

  5. sql子查询 嵌套SELECT语句

    嵌套SELECT语句也叫子查询,一个 SELECT 语句的查询结果能够作为另一个语句的输入值.子查询不但能够出现在Where子句中,也能够出现在from子句中,作为一个临时表使用,也能够出现在sele ...

  6. MySQL锁类型以及子查询锁表问题、解锁

    MySQL中select * for update锁表的范围 MySQL中select * for update锁表的问题 由于InnoDB预设是Row-Level Lock,所以只有「明确」的指定主 ...

  7. Mysql查询优化器之关于子查询的优化

    下面这些sql都含有子查询: mysql> select * from t1 where a in (select a from t2); mysql> select * from (se ...

  8. Mysql - 性能优化之子查询

    记得在做项目的时候, 听到过一句话, 尽量不要使用子查询, 那么这一篇就来看一下, 这句话是否是正确的. 那在这之前, 需要介绍一些概念性东西和mysql对语句的大致处理. 当Mysql Server ...

  9. MySQL 使用JOIN优化子查询

    1.数据准备 mysql> select * from student; +----+--------+----------+---------+-------------+ | id | na ...

随机推荐

  1. idea 出现 bootstrap.properties 中的内容不能识别

    错误截图如下 依赖库引用问题 第一种: <dependency> <groupId>org.springframework.cloud</groupId> < ...

  2. Win64 驱动内核编程-33.枚举与删除对象回调

    转载:http://www.voidcn.com/article/p-wulgeluy-bao.html 枚举与删除对象回调 对象回调存储在对应对象结构体里,简单来说,就是存储在 ObjectType ...

  3. 使用 Nginx 阻止恶意 IP 访问

    找到具有明显特征的访问记录,比如: /Dec/::: +] "-" "Ouija_x.86/2.0" "-" 也许是某个开源框架的漏洞,执行 ...

  4. Eclipse使用Working Set

    当Eclipse中创建了太多的project,太多了,看的眼花缭乱,不好管理,也不想更换工作空间,Eclipse中 Java Working Set 工作集,可以将这些project分组,就像文件夹分 ...

  5. Spring-boot2X基于sharding-jdbc3.x分表分库

    ShardingSphere是一套开源的分布式数据库中间件解决方案组成的生态圈,它由Sharding-JDBC.Sharding-Proxy和Sharding-Sidecar(计划中)这3款相互独立的 ...

  6. Git 克隆

    工作中大家分工协作,共同开发维护一个项目,git仓库放在公共的服务器上,如github.Gitee等. 在这种情况使用git,需要先克隆仓库到本地. 克隆非常简单,使用命令git clone 将远程仓 ...

  7. Mysql 添加用户并授所有权

    创建用户并授权GRANT ALL PRIVILEGES ON *.* TO 'caoxiaobo'@'%' IDENTIFIED BY 'Caoxiaobo0917!' WITH GRANT OPTI ...

  8. 关于UiAutomator无法识别的元素

    1.关于没有name,没有ID的元素的定位---通用篇解题思路:因为没有name,id:其实剩下的选择已不多,要么xpath,要么className.xpath木有好印象(稳定性不高,加之1.0x后需 ...

  9. 通过fsockopen()方法从中国福彩网获取双色球历史中奖数据

    # 以下代码基于 CI 框架 # public function history_draw($page = 0) { set_time_limit(0); $page++; $up2now = dat ...

  10. 递归实现全排列python

    python递归实现"abcd"字符串全排列 1.保持a不动,动bcd 2.保持b不动,动cd 3.保持c不动,动d def pailie(head="",st ...