关于非相关子查询改写经验

1、问题

最近又又又再项目中遇到这样奇奇怪怪的写法

c 复制代码
select count(1)
  from t1
 where exists (select 1
          from (select case t1.state when 1 
          then sum(sum(t3.ll*0.5)+nvl(max(t2.ll),0)) over(partition by t3.c1) end as bstate
                       from t2,
                            t3
                      where t2.id=t3.c1 and t2.id=t1.id
                   group by t3.c1)
         where bstate>10 );

计划:

从计划上看只能做非相关子查询,即每次获取主查询一条记录去子查询根据条件判断过滤获取结果。为什么会这样?原因是因为exists中select存在主查询条件作为查询项得出结果后还要通过where去过滤,因此只能主查询一条一条去判断,主查询数据量增加,效率急剧下降。这种写法需要转换成相关子查询,即做成相关关联,将子查询做成结果集与主查询关联。

思路拆解

获取子查询结果集,

子查询查询项

sum(sum(t3.ll*0.5)+nvl(max(t2.ll),0)) over(partition by t3.c1)是子查询中获取的

因此可以先把这一部分求出来

c 复制代码
select t3.c1,sum(t3.ll*0.5)+nvl(max(t2.ll),0) as bstate
      from t2, t3 where t2.id=t3.c1 
group by t3.c1

然后与主查询关联条件是t2.id=t1.id,而t2.id=t3.c1那么t3.c1=t1.id

Exists:是判断子查询中是否存在与主查询相匹配的结果。如果相匹配就返回主查询相匹配的结果。它只是判断主查询是否有相匹配的记录,因此exsits只做判断,如果转换成inner join,inner join是获取相匹配的记录,如果子查询结果有重复,与主查询做关联,那会重复输出结果,因此需要子查询去重才能成功转换成inner join。而group by t3.c1(关联列)达到去重效果。

因此可以改写成

c 复制代码
select t1.*,case t1.state when 1 then bstate end bstate
           from t1,
(select t3.c1,sum(t3.ll*0.5)+nvl(max(t2.ll),0) as bstate
      from t2, t3 where t2.id=t3.c1 
group by t3.c1) b where t1.id=b.c1

然后再把查询项获取出来再where处理,最终完成改写。

2、改写

c 复制代码
select count(1)  from (
select t1.*,case t1.state when 1 then bstate end bstate
           from t1,
(select t3.c1,sum(t3.ll*0.5)+nvl(max(t2.ll),0) as bstate
      from t2, t3 where t2.id=t3.c1 
group by t3.c1) b where t1.id=b.c1 ) tt
where bstate>10;

计划:

计划上达到预期的效果,做成关联。

执行时间从

下降至

3、小结

对于这种写法,主要思路是拆解,子查询能获取的先获取,然后找与主查询的关联关系关联主查询,最后做条件过滤。

相关推荐
别叫我老干部几秒前
博客 / CMS 数据库表结构设计
sql
计算机安禾11 分钟前
【数据库系统原理】第15篇:范式理论(上):1NF至BCNF——消除非主属性对码的传递依赖与部分依赖
数据库
你的保护色28 分钟前
数据库第一章-基础知识学习
数据库·学习
倔强的石头_32 分钟前
《Kingbase护城河》——数据库卡顿急救手册:会话状态深度解析与“僵尸进程”排查实战
数据库
ManageEngine卓豪1 小时前
数据库可观测性:MySQL与Redis监控核心监控指标与全栈运维解决方案
数据库·redis·mysql·数据库性能·数据库监控
真实的菜1 小时前
Redis 从入门到精通(十四):Redis 7.x 新特性全解 —— 系列收官之作
数据库·redis·缓存
哭哭啼1 小时前
pgSql 事务篇
java·数据库·postgresql
霸道流氓气质1 小时前
从MySQL到云原生:全面解析阿里云PolarDB数据库及其与MySQL的核心差异
数据库·mysql·云原生
这个DBA有点耶2 小时前
时序数据库选型:吞吐、压缩与查询延迟的均衡之术
数据库·sql·架构·时序数据库·dba