[记录]一个30秒的sql,我是怎么把它改到0.5秒的

一天下午正在愉快的写代码,组长走过来打开了系统的一个功能,跟我说这个玩意查询要18秒,忍受不了啊,最主要它还引发了其它问题,所以让我想办法解决,那我就排查一下吧.

直接从前端找到接口,定位到相应的接口,这个接口很简单里面就一些简单逻辑然后调用mapper层的一个方法最终返回数据给前端,那这样直接拿这个sql跑一下就知道了

把where条件给干掉之后,limit100条先看看什么情况先

好家伙给干出了34秒,妥妥的慢sql了,先看看具体的sql

sql 复制代码
SELECT 
DISTINCT
  i.*,
  csc.suit_sku_image,
  (
    SELECT
      string_agg (sort_name, ',')
    FROM
      commodity_info_sort k
      JOIN commodity_sort k1 ON k.sort_id = k1.id
    WHERE
      k.info_id = i.id
  ) AS "type",
  (
    SELECT
      SUM(CASE WHEN csgd.sales_type = '1' THEN csgd.amount ELSE 0 END)
    FROM
      commodity_store_goods_detail csgd
    WHERE
      csgd.info_id = i.id
  ) AS "total_amount_sale_type_1",
  (
    SELECT
      SUM(CASE WHEN csgd.sales_type = '0' THEN csgd.amount ELSE 0 END)
    FROM
      commodity_store_goods_detail csgd
    WHERE
      csgd.info_id = i.id
  ) AS "total_amount_sale_type_0",
  (SELECT ci.name FROM commodity_info ci WHERE ci.id = i.from_where) AS "infoname",
  (SELECT ci.main_image FROM commodity_info ci WHERE ci.id = i.from_where) AS "infomainImage"
FROM
  commodity_info i
  LEFT JOIN commodity_package_relationship cpr ON cpr.info_id = i.id
  LEFT JOIN commodity_suit_compose csc ON csc.only_info_id = i.id
-- limit 10

一眼看过去已经觉得这个sql绝对不会快了,有 DISTINCT ,有5个子查询,其实有两个还是针对该sql的主表的,不管三七二十一跑个执行计划看看我们用的是pgsql所以我是这样跑的

用关键字 EXPLAIN

优先看有没有 Scan on 全表扫描的

根据上图的执行计划可以看到这三张表都走了全表扫描,原以为问题就在这里了,但是一看消耗才这么一点,那肯定不是这个了,继续往下看,再看 相关子查询 SubPlan,不看不知道一看吓一跳

第一个子查询

我们根据第一行 Aggregate可以看出 每次返回 1行(rows=1)消耗时间0.669毫秒(actual time=0.669)姑且算0.7毫秒,然后

loops=11765 也就是被执行了11765次,一次就是0.7毫秒,11765次就是11765x0.7=8,235.5毫秒

换算成秒也就是8秒钟,一个子查询就8秒钟了能不慢嘛.

问题来了,为什么慢呢?

往下看 ,嵌套循环连接 ,条件是 k.sort_id = k1.id

-> Nested Loop (cost=0.00..218.70 rows=1 width=13) (actual time=0.432..0.668 rows=1 loops=11765)

Join Filter: (k.sort_id = k1.id)

Rows Removed by Join Filter: 29

排除了29条数据这里看不出太多东西不过也可以根据Join Filter看出这个连接没有走索引

继续往下看,重点来了

-> Seq Scan on commodity_info_sort k (cost=0.00..210.72 rows=1 width=4) (actual time=0.426..0.660 rows=1 loops=11765)

Filter: (info_id = i.id)

Rows Removed by Filter: 11518

Seq Scan on全表扫描,条件是info_id = i.id

说明一下i.id是最外层的主表的id,

意味着外层主表的每一行数据都跟这个子查询里面的每一行数据都去查询了是不是=.

所以他的查询次数就变成N × M

id是天然的索引,这里没有走索引的话可以看看commodity_info_sort的 info_id 字段是不是没有创建索引.创建了索引我感觉这里会快一大半

Rows Removed by Filter: 11518 这个意味着在扫描的过程中有11518是不符合要求被过滤掉的,说明每次扫描读取了很多不相关行,每次都要遍历大量行,只留下极少数匹配,重复 11,765 次 极其低效

到现在答案呼之欲出了,下面的几个子查询也是如出一辙,都是耗时大户


2025.10.28排查记录先写到这里,后面花时间继续写改造这个sql的过程

相关推荐
折翼的恶魔5 小时前
SQL187 每份试卷每月作答数和截止当月的作答总数。
sql
行者游学5 小时前
ETCD 集群备份与恢复
数据库·etcd
行者游学5 小时前
ETCD 学习使用
数据库·学习·etcd
代码程序猿RIP5 小时前
【Etcd 】Etcd 详解以及安装教程
linux·数据库·etcd
tqs_123455 小时前
创建es索引
数据库·es
间彧6 小时前
Mysql优化指南:善用覆盖索引,提升查询效率
数据库
间彧6 小时前
MySQL优化指南:执行计划type字段详解
数据库
ximy13356 小时前
Mysql基础知识之SQL语句——库表管理操作
sql·mysql·oracle