mysql怎样优化count(*) from 表名 where …… or ……这种慢sql

一 问题描述

线上发现一条类似这样的慢sql(查询时长8s):

select id,name,(select count(*) from t14 where t14.id=t15.id or t14.id2=t15.id) as cnt

from t15 ;

t14的id和id2字段上都有索引,但是因为条件里有or,导致走的是全表扫描:

如果没用count(*),而是select 字段这种方式,那可以用union这种方式替代or,但这里是count(),则有些不同。

二 优化逻辑

将select count(*) from t14 where t14.id=t15.id or t14.id2=t15.id改为以下三种情况:

1、 t14.id=t14.id2,此时t14.id = t15.id与t14.id2=t15.id是等价的,写哪个都可以

2、 t14.id!=t14.id2时,分成两种情况:

t14.id = t15.id

② t14.id2 = t15.id

三 改写后的sql

select id,name,

(select count(*) from t14 where t14.id=t15.id and t14.id=t14.id2)

(select count(*) from t14 where t14.id=t15.id and ifnull(t14.id,'isnull')!=ifnull(t14.id2,'isnull')

)

(select count(*) from t14 where t14.id2=t15.id and t14.id!=t14.id2)

as cnt

from t15

加ifnull(字段,'isnull')函数是因为发现关联字段是null时,关联不上,所以这里将这些空值转换为了isnull这个字符串。

执行计划走了索引:

逻辑看起来比之前复杂了,但是查询时长由8秒降到了1.2秒

相关推荐
muddjsv9 小时前
SQL 最常用技能详解与实战示例
数据库·sql·mysql
ᰔᩚ. 一怀明月ꦿ10 小时前
MySQL 学习目标
学习·mysql·adb
他们叫我阿冠14 小时前
Day4学习--MySQL高级
数据库·学习·mysql
罗超驿15 小时前
20.MySQL事务隔离级别示例详解(脏读、不可重复读、幻读)
java·数据库·mysql·面试
独泪了无痕16 小时前
MySQL中 JSON 数据类型使用指南
mysql
我是一颗柠檬16 小时前
【MySQL全面教学】MySQL基础与环境搭建Day1(2026年)
数据库·后端·sql·mysql·database
我是一颗柠檬17 小时前
【MySQL全面教学】MySQL数据类型详解Day2(2026年)
数据库·后端·sql·mysql·database
小江的记录本17 小时前
【Java并发编程】锁机制:volatile:JMM内存模型、可见性/禁止指令重排、内存屏障、单例模式中的应用(附《思维导图》+《面试高频考点清单》)
java·后端·python·mysql·单例模式·面试·职场和发展
身如柳絮随风扬17 小时前
CentOS 7 搭建 MySQL 主从复制集群:从零到生产级高可用
linux·mysql·centos
数据库小学妹17 小时前
MySQL 性能监控实战:从零搭建 Prometheus + Grafana 监控告警体系(附排查 SOP)
mysql·性能优化·grafana·prometheus·dba