博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
SQL优化(SQL TUNING)可大幅提升性能的实战技巧之一——让计划沿着索引跑
阅读量:5059 次
发布时间:2019-06-12

本文共 975 字,大约阅读时间需要 3 分钟。

我们进行SQL优化时,经常会碰到对大量数据集进行排序,然后从排序后的集合取前部分结果的需求,这种情况下,当我们按照常规思路去写SQL时,系统会先读取过滤获得所有集合,然后进行排序,再从排序结果取出极少量结果,这个过程中,大量数据的扫描读取、过滤、排序会消耗掉大量的系统资源,SQL性能也会存在很大的问题,实践中,几分钟乃至几个小时不出结果的情况很常见。为了优化这种场景的SQL,我们经常会让查询顺序扫描建在排序列上的索引,已避开大量的数据读取和排序。

但实践中发现,当索引列不在条件中出现时,ORACLE不会产生扫描索引的计划,即使用hint也不能让查询沿着目的索引扫描,例如:

create table t1(c1 int,c2 char(10));

create index idx1_t1 on t1(c1);

select * from (

select  * from t1 order by c1)

where rownum<6;

很明显,这种写法会导致先读取表,再进行排序,然后取前5条记录,计划如下:

如果我们这么写,语义是一样的,但会省去了大量的读取和排序代价:

select /*+ index(t1,idx1_t1)*/* from t1 where rownum<6;

该SQL的执行计划如下:

大家可以看到,该SQL并没按照hint指示,顺序扫描idx1_t1索引,这是为什么呢?个人猜测,可能是Oracle优化器认为过滤条件内没c1列,走索引比走FTS效率更低,所以,干脆就不考虑走索引这种计划,即使用hint也要忽略,这点感觉有点不尽人意。那么,我们想什么办法才能让优化器选择扫描idx1_t1的计划呢?我们只需要在where中加个c1列上的条件就可以了,例如:

select /*+ index(t1,idx1_t1)*/* from t1 where rownum<6 and c1<>-1;

这么修改后的计划如下:

由此可见,我们优化类似场景时,只需满足两点:

1、排序列上存在索引;

2、where条件中有该索引列上的条件;

如果能实现按照索引扫描,性能有成千上万倍的提升也是非常可能的,这点在实践中得到了验证。

 

转载于:https://www.cnblogs.com/lhdz_bj/p/8846936.html

你可能感兴趣的文章
Python图像处理库(PIL)
查看>>
cocos2d-x载入图以及后前台切换
查看>>
数字处理类
查看>>
互观与评价
查看>>
验证视图状态 MAC 失败的解决办法
查看>>
css整理-01选择器和继承
查看>>
webapp前端开发总结
查看>>
【MAC下学习Unix网络编程】第一个例子中解决unp.h 在mac下的编译问题
查看>>
jquery ztree插件
查看>>
Java finally关键字
查看>>
InstallShield Limited Edition for Visual Studio 2013 图文教程(教你如何打包.NET程序)
查看>>
LUA安装过程
查看>>
Python-aiohttp百万并发
查看>>
牛顿法求平方根及习题1.6-1.8
查看>>
电赛初探(二)——语音采集回放系统
查看>>
SQL SERVER 如何调试存储过程
查看>>
php修改和增加xml结点属性
查看>>
Mysql插入数据是问号的乱码
查看>>
设计模式之原型模式
查看>>
(转)页面滚动图片加载
查看>>