设为首页 收藏本站
查看: 1030|回复: 0

[经验分享] SQL语句优化规律总结(ORACLE)

[复制链接]

尚未签到

发表于 2016-7-24 11:57:49 | 显示全部楼层 |阅读模式
1、FROM:
ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,因此FROM子句中写在最后的表(基础表 driving table)将被最先处理. 在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表(放在from的最后) 。如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表.
2、WHERE:
ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾。
3、在where子句中使用is null或is not null的语句优化器是不允许使用索引的,尽量不用。
4、少使用或不使用NOT。
5、如果通配符(%)在搜寻词首出现, Oracle系统不使用此列做索引。在很多情况下可能无法避免这种情况,但是一定要心中有底,通配符如此使用会降低查询速度。然而当通配符出现在字符串其他位置时,优化器就能利用索引。在下面的查询中索引得到了使用
select * from employee where last_name like 'c%';
6、 SELECT子句中避免使用 ‘ * ‘
7、避免在索引列上使用计算.如(WHERE SAL  > 25000/12)
8、任何在Order by语句的非索引项或者有计算表达式都将降低查询速度。
9、关于IN、exists和DISTINCT
第一种格式是使用IN操作符:
... where column in(select * from ... where ...);

第二种格式是使用EXIST操作符:
... where exists (select 'X' from ...where ...);

第二种格式要远比第一种格式的效率高。

通常来说 , 采用表连接(emp.depart_cod=depart.depart_code)的方式比EXISTS更有效率
所以:连接>exists>IN。
用EXISTS替换DISTINCT
当提交一个包含一对多表信息(比如部门表和雇员表)的查询时,避免在SELECT子句中使用DISTINCT. 一般可以考虑用EXIST替换

低效:
    SELECT DISTINCT DEPT_NO,DEPT_NAME
    FROM DEPT D,EMP E
    WHERE D.DEPT_NO = E.DEPT_NO

高效:
    SELECT DEPT_NO,DEPT_NAME
    FROM DEPT D
    WHERE EXISTS ( SELECT ‘X’
                    FROM EMP E
                    WHERE E.DEPT_NO = D.DEPT_NO);
10、下面的查询将强迫对orders表执行顺序操作:
SELECT * FROM orders WHERE (customer_num=104 AND order_num>1001) OR order_num=1008
虽然在customer_num和order_num上建有索引,但是在上面的语句中优化器还是使用顺序存取路径扫描整个表。因为这个语句要检索的是分离的行的集合,所以应该改为如下语句:
SELECT* FROM orders WHERE customer_num=104 AND order_num>1001
UNION
SELECT * FROM orders WHERE order_num=1008
这样就能利用索引路径处理查询。
11、避免相关子查询
一个列的标签同时在主查询和where子句中的查询中出现,那么很可能当主查询中的列值改变之后,子查询必须重新查询一次。查询嵌套层次越多,效率越低,因此应当尽量避免子查询。如果子查询不可避免,那么要在子查询中过滤掉尽可能多的行。
12、在缺省情况下,ORACLE采用CHOOSE优化器, 为了避免那些不必要的全表扫描(full table scan) , 你必须尽量避免使用CHOOSE优化器,而直接采用基于规则或者基于成本的优化器.
13、一般情况下性能上:count(一个索引字段) > count(*) > count(一个非索引字段)
14、在select和where中尽量不要使用sysdate函数;因为它是每处理一行数据就要到系统中取一次系统时间;一般的解决的方法先把系统时间取出来当做一个变量传入:select sysdate from dual;
15.CHAR的长度是固定的,而VARCHAR2的长度是可以变化的;
     CHAR的效率比VARCHAR2的效率稍高;
     但VARCHAR2比CHAR节省磁盘空间。如磁盘空间节省的多;有时VARCHAR2也会比CHAR的效率稍高;看实际情况而定。
16.数字 — 长整型 Number(11)
  数字 — 字节型 Number(3)
  数字 — 整型 Number(5)
  日期/时间 Date Date
  布尔型 Number(1) 或 char(1)
精度型 一般是 Number(M,N),M是有效数字,N是小数点后的位数(默认0)
Number默认是Number(M,N); M=38,可以根据数字随机适应;但占用磁盘空间大。
17.nchar,nvarchar2,nclob和char:varchar2, clob:
在utf-8的oracle数据库时;最大char(4000);nchar(2000);
                                 但一个汉字char(3);nchar(1);
所以在汉字用的多和频繁的文本时;建议用nchar();其它的clob;varchar2和char相似。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-248608-1-1.html 上篇帖子: Oracle中数据字典的使用方法 下篇帖子: Oracle 查询表空间使用情况(转)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表