oracle数据库sql的优化总结_oracle

一:使用where少使用having;

二:查两张以上表时,把记录少的放在右边;

三:减少对表的访问次数;

四:有where子查询时,子查询放在最前;

五:select语句中尽量避免使用*(执行时会把*依次转换为列名);

六:尽量多的使用commit;

七:Decode可以避免重复扫描相同的记录或重复连接相同的表;

八:通过内部函数也可提高sql效率;

九:连接多个表时,使用别名并把别名前缀于每个字段上;

十:用exists代替in

十一:not exists代替 not in(not in 字句将执行一个内部的排序和合并,任何情况下,not in是最低效的,子查询中全表扫描了。为了避免使用not in,可以改写成outer joins或not exists);

十二:表连接比exists更高效;

十三:用exists替换distinct

       例: 

             低:                                                                    高:

             select distinct dept_no, dept_name                        select dept_no, dept_name

               from dept d, emp e                                               from dept d

            where d.dept_no = e.dept_no;                               where exists (select 1 from emp e where e.dept_no = d.dept_no);

十四:使用TKPROF工具来查询sql性能状态;

十五:用索引提高效率(代价是:索引需要空间,而且定期重构索引很有必要:ALTER INDEX<INDEXNAME> REBUILD<TABLESPACENAME);

先介绍下索引的原理,方便接下来对索引的优化的理解:

     通过索引找到rowid,然后通过rowid访问表。但如果查询的列包括在index中,将不在执行第二部操作,因为检索数据保存在索引中,单单访问索引就可以完全满足查询要求。

前提提要:在十六例中,LODGING列有唯一索引;MANAGER列上有非唯一性索引。

十六:索引范围查询(INDEX RANGE SACEN):

        适用于两种情况:

        1)基于一个范围的查询:

              SELECT LODGING FROM LODGING WHERE LODGING LIKE 'M%'

        (where字句条件包括一系列的值,oracle将通过索引范围查询方式查询LODGING_PK)

        2) 基于非唯一性索引的检索:

              SELECT LODGING FROM LODGING WHERE MANAGER = 'LI';

         (此查询分两步:LODGING$MANAGER的索引范围查询得到所有符合条件记录的rowid,然后通过rowid访问表得到LODGING列的值。该索引为非唯一性索            引,数据库不能对它执行索引唯一扫描)

where字句中,如果索引列所对应的值的第一个字符由通配符开始,索引将不被采用,而会全表扫描,如 SELECT..... WHERE MANAGER LIKE '%LI'

十七:基础表的选择:

        基础表:最先访问的表(通常以全表扫描的方式被访问)。

        根据优化器的不同,SQL语句中基础表的选择是不一样的:

            如果使用CBO,优化器会检查SQL语句中的每个表的物理大小,索引的状态,然后选用话费最低的路径。

            如果使用RBO,并且所有的连接条件都有索引对应,这种情况下基础表就是FROM字句中列在最后的表

        例:

             SELECT A.NAME, B.MANAGER FROM WOKER A, LODGING B WHERE A.LODGING = B.LODGING;

             由于LODGING列上有一个索引,而且WORKER表中没有相比较的索引,WORKER表将被作为查询基础表。

十八:多个平等的索引:

        当SQL语句的执行路径可以使用分布在多个表上的多个索引时,oracle会同事使用多个索引并在运行时对它们的记录合并,检索仅对全部索引有效的记录。

        oracle选择执行路径是,唯一索引等级高于非唯一索引,只有当where字句中索引列和常量比较才有效。如果索引列和其它表的索引列相比较,这种字句在优化器中等级非常低;

        如果不同表中两个相同等级的索引将被引用,根据FROM字句中表的顺序决定哪个先被使用。FROM字句中最后的表索引优先级高。如果相同表中两个相同等级的索引将被引用,where字句中最先被引用的索引将有最高的优先级。

       例:DEPTNO上有非唯一性索引,EMP_CAT也有非唯一性索引

            SELECT ENAME FROM EMP WHERE DEPT_NO = 20 AND EMP_CAT = 'A'; 

            DEPTNO索引将被先检索,然后同EMP_CAT索引检索出的结果合并,执行路径如下:

              TABLE ACCESS BY ROWID ON EMP

              AND _EQUAL

              INDEX RANGE SCAN ON  DEPT_IDX

              INDEX RANGE SCAN ON CAT_IDX

十九:等式比较与范围比较:

       先上例子:

           SELECT ENAME FROM EMP WHERE DEPT_NO > 20 AND EMP_CAT = 'A';

         (在两个非唯一性索引前提下)此时范围索引不被使用,通过EMP_CAT索引查询出记录再与DEPT_NO条件进行比较

      注意:唯一性所以做范围比较时,等级要比非唯一性索引的等式比较低;

二十:强制索引失效:

        如果两个或两个以上索引具有相同的等级,可以强制命令oracle优化器使用其中的一个。 那何时使用此种策略呢?如果一个索引已接近于唯一,而另一索引有很多重复的值,排序与合并反而会成为负担,此时可以屏蔽后者使其索引失效。

       (失效方式:对索引列加入计算'+0'或'||""');

时间: 2024-09-19 23:53:11

oracle数据库sql的优化总结_oracle的相关文章

oracle数据库sql的优化总结

  自己对oracle sql的一些优化总结,自己也记录下来,也希望对大家有帮助: 一:使用where少使用having; 二:查两张以上表时,把记录少的放在右边; 三:减少对表的访问次数; 四:有where子查询时,子查询放在最前; 五:select语句中尽量避免使用*(执行时会把*依次转换为列名); 六:尽量多的使用commit; 七:Decode可以避免重复扫描相同的记录或重复连接相同的表; 八:通过内部函数也可提高sql效率; 九:连接多个表时,使用别名并把别名前缀于每个字段上; 十:用

Oracle数据库调试和优化详解

此文主要是关于Oracle数据库调试和优化方面的相关内容的介绍,还有涉及到Oracle数据库中命里率的相关问题的解答,其中包括不同的算法之间性能的比对.以下就是相关内容的介绍. 关于Oracle中各个命中率的计算以及相关的调优 1)Library Cache的命中率: 计算公式: 1. Library Cache Hit Ratio = sum(pinhits) / sum(pins) 2. SQL>SELECT SUM(pinhits)/sum(pins) FROM V$LIBRARYCACH

oracle 数据库 sql 问题

问题描述 oracle 数据库 sql 问题 select pdept, deptsort, deptlevel, 省略 from (省略) t group by cube(pdept, deptsort, deptlevel) -- having grouping_id(pdept, deptsort, deptlevel) not in(2)------------------------A 1.问题是这样的 查询数据可能有 0 1 2 3 4 5 6 7 2.去掉 A处的注释,not in

sql优化-mysql数据库sql语句优化,求大神!!!!

问题描述 mysql数据库sql语句优化,求大神!!!! SELECT DISTINCT uid, level,username,ansnum FROM test WHERE level=100 GROUP BY uid ORDER BY ansnum DESC LIMIT 12; uid.ansnum均已建索引,主要是GROUP BY uid导致特别慢,如何提速??? 解决方案 MySQL数据库SQL语句优化原则 解决方案二: 根据你的查询需求,没有特别好的优化办法.注意group by 和o

如何保持Oracle数据库SQL性能的稳定性

如何保持Oracle数据库SQL性能的稳定性 >     

对学Oracle数据库初学者的开场篇_oracle

前言:因为项目原因,近期开始学习Oracle数据库.Oracle是目前最流行的数据库之一,功能强大,性能卓越,相对的学习的难度还是不小.我打算将自己的学习过程记录下来,做个积累,方便自己和其他的学习者. 写在学习前面的话,主要是: 1.把一些常见问题的问答分享一哈: 2.对学习路线的一个安排: 3.预期的结果展望. 1.Oracle数据库学习常见问题问答 Question1: Oracle数据库的书很难看懂,oracle真的很难学习吗? Answer1: Oracle就是一个小软件,它把复杂的事

Oracle数据库的空间管理技巧_oracle

正在看的ORACLE教程是:Oracle数据库的空间管理技巧. 在Oracle数据库中,DBA可以通过观测一定的表或视图来了解当前空间的使用状况,进而作出可能的调整决定.  一.表空间的自由空间  通过对表空间的自由空间的观察,可用来判断分配给某个表空间的空间是太多还是不够.请看下列的语句  可以看出,在FileNo为12的表空间RBS中,只有0.19%的分配空间未被使用,这个比例太小了,而在SYSTEM及TEMP等表空间中,高达80%以上的空间未被利用,对于生产型数据库,这个表空间的设置有些偏

在Linux系统上同时监控多个Oracle数据库表空间的方法_oracle

一,设计背景  由于所在公司ORACLE数据库较多,传统人工监控表空间的方式较耗时,且无法记录历史表空间数据,无法判断每日表空间增长量,在没有gridcontrol/cloudcontrol软件的情况下,笔者设计如下表空间监控方案,大家也可以根据自己的实际情况对下面的方案进行修改.二,设计思路 通过dblink将来查询到的表空间数据集中汇总到一张表里通过crontab跑定时任务从各台服务器获取表空间使用情况信息.三,具体实施步骤  1.所在oracle数据库ip地址信息(下面为举例说明具体情况要

ORACLE数据库日常维护知识点总结_oracle

首先要说的是,不同版本数据库提供的系统表会有不同,你可以根据数据字典查看该版本数据库所提供的表 like this: select * from dict where table_name like '%SESSION% '; 就可以查出一些表,然后根据这些表就可以获得会话信息. 像这样就是 查询当前正在操作的会话: SELECT SID, SERIAL#, STATUS, USERNAME, SCHEMANAME, OSUSER,TERMINAL, MACHINE, PROGRAM, A.NA