《Oracle SQL性能优化方法研究》由会员分享,可在线阅读,更多相关《Oracle SQL性能优化方法研究(16页珍藏版)》请在金锄头文库上搜索。
1、OracleSQL性能优化方法探讨Oracle性能优化方法(SQL篇)11综述22表分区的应用23访问Table的方式34共享SQL语句35选择最有效率的表名顺序56WHERE子句中的连接顺序67SELECT子句中避免使用*68减少访问数据库的次数69使用DECODE函数来减少处理时间710整合简单,无关联的数据库访问811删除重复记录812用TRUNCATE替代DELETE913尽量多使用COMMIT914计算记录条数915用Where子句替换HAVING子句916减少对表的查询1017通过内部函数提高SQL效率.1118使用表的别名(Alias)1219用EXISTS替代IN1220用NO
2、T EXISTS替代NOT IN1321识别低效执行的SQL语句1322使用TKPROF 工具来查询SQL性能状态1423用EXPLAIN PLAN 分析SQL语句1424实时批量的处理161 综述ORACLE数据库的性能调整是个重要,却又有难度的话题,如何有效地进行调整,需要经过反反复复的过程。在数据库建立时,就能根据应用的需要合理设计分配表空间以及存储参数、内存使用初始化参数,对以后的数据库性能有很大的益处,建立好后,又需要在应用中不断进行应用程序的优化和调整,这需要在大量的实践工作中不断地积累经验,从而更好地进行数据库的调优。数据库性能调优的方法l 调整内存l 调整I/Ol 调整资源的争
3、用问题l 调整操作系统参数l 调整数据库的设计l 调整应用程序本文针对应用程序的调整,来说明对数据库性能如何进行优化。2 表分区的应用对于海量数据的表,可以考虑建立分区以提高操作效率。建立分区一般以关键字为分区的标志,也可以以其他字段作为分区的标志,但效率不如关键字高。建立分区的语句在建表时可以进行说明:create table TABLENAME()partition by range (PutOutNo)(partition PART1 values lessthan (200312319999)partition PART2 values lessthan (200412319999)T
4、ABLENAME2003Data2005Data2004Data。partition PART3 values lessthan (200512319999)。建好分区后,数据的逻辑存储方式进行了优化这样,在进行大部分数据查询,数据更新和数据插入时,Oracle自动判断操作应该在哪个分区进行,避免了整表操作,提高了执行的效率3 访问Table的方式ORACLE 采用两种访问表中记录的方式: l 全表扫描 全表扫描就是顺序地访问表中每条记录. ORACLE采用一次读入多个数据块(database block)的方式优化全表扫描. l 通过ROWID访问表 可以采用基于ROWID的访问方式情况,提
5、高访问表的效率, , ROWID包含了表中记录的物理位置信息.ORACLE采用索引(INDEX)实现了数据和存放数据的物理位置(ROWID)之间的联系. 通常索引提供了快速访问ROWID的方法,因此那些基于索引列的查询就可以得到性能上的提高. 4 共享SQL语句 为了不重复解析相同的SQL语句,在第一次解析之后, ORACLE将SQL语句存放在内存中.这块位于系统全局区域SGA(system global area)的共享池(shared buffer pool)中的内存可以被所有的数据库用户共享. 因此,当执行一个SQL语句(有时被称为一个游标)时,如果它和之前的执行过的语句完全相同, OR
6、ACLE就能很快获得已经被解析的语句以及最好的执行路径. ORACLE的这个功能大大地提高了SQL的执行性能并节省了内存的使用.可是ORACLE只对简单的表提供高速缓冲(cache buffering) ,这个功能并不适用于多表连接查询. 数据库管理员必须在init.ora中为这个区域设置合适的参数,当这个内存区域越大,就可以保留更多的语句,当然被共享的可能性也就越大了. 当向ORACLE 提交一个SQL语句,ORACLE会首先在这块内存中查找相同的语句.这里需要注明的是,ORACLE对两者采取的是一种严格匹配,要达成共享,SQL语句必须完全相同(包括空格,换行等). 共享的语句必须满足三个条
7、件: l 字符级的比较: 当前被执行的语句和共享池中的语句必须完全相同.例如: SELECT * FROM EMP; 和下列每一个都不同 SELECT * from EMP; Select * From Emp; SELECT * FROM EMP; l 两个语句所指的对象必须完全相同: 例如: 用户 对象名 如何访问 Jack sal_limit private synonymWork_city public synonymPlant_detail public synonymJill sal_limit private synonymWork_city public synonymPlan
8、t_detail table owner下列SQL语句不能在这两个用户之间共享. select max(sal_cap) from sal_limit; 原因 每个用户都有一个private synonym - sal_limit , 它们是不同的对象 下列SQL语句能在这两个用户之间共享. select count(*) from work_city where sdesc like NEW%; 原因:两个用户访问相同的对象public synonym - work_city 下列SQL语句不能在这两个用户之间共享. select a.sdesc,b.location from work_c
9、ity a , plant_detail b where a.city_id = b.city_id 原因:用户jack 通过private synonym访问plant_detail 而jill 是表的所有者,对象不同. l 两个SQL语句中必须使用相同的名字的绑定变量(bind variables) 例如:第一组的两个SQL语句是相同的(可以共享),而第二组中的两个语句是不同的(即使在运行时,赋于不同的绑定变量相同的值) 1.select pin , name from people where pin = :blk1.pin; select pin , name from people
10、where pin = :blk1.pin; 2.select pin , name from people where pin = :blk1.ot_ind; select pin , name from people where pin = :blk1.ov_ind;5 选择最有效率的表名顺序ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,因此FROM子句中写在最后的表(基础表 driving table)将被最先处理. 在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.当ORACLE处理多个表时, 会运用排序及合并的方式连接它们.首先,扫描第一个表
11、(FROM子句中最后的那个表)并对记录进行派序,然后扫描第二个表(FROM子句中最后第二个表),最后将所有从第二个表中检索出的记录与第一个表中合适记录进行合并. 例如: 表 TAB1 16,384 条记录,表 TAB2 1 条记录 选择TAB2作为基础表 (最好的方法) select count(*) from tab1,tab2选择TAB2作为基础表 (不佳的方法) select count(*) from tab2,tab1如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表. 例如: EMP表描述了LOCA
12、TION表和CATEGORY表的交集. SELECT * FROM LOCATION L , CATEGORY C, EMP E WHERE E.EMP_NO BETWEEN 1000 AND 2000 AND E.CAT_NO = C.CAT_NO AND E.LOCN = L.LOCN 将比下列SQL更有效率 SELECT * FROM EMP E , LOCATION L , CATEGORY C WHERE E.CAT_NO = C.CAT_NO AND E.LOCN = L.LOCN AND E.EMP_NO BETWEEN 1000 AND 2000 6 WHERE子句中的连接顺序
13、 ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾.例如:(低效)SELECT FROM EMP E WHERE SAL 50000 AND JOB = MANAGER AND 25 (SELECT COUNT(*) FROM EMP WHERE MGR=E.EMPNO); (高效)SELECT FROM EMP E WHERE 25 50000 AND JOB = MANAGER;7 SELECT子句中避免使用*当在SELECT子句中列出所有的COLUMN时,使用动态SQ
14、L列引用 * 是一个方便的方法.可是,这是一个非常低效的方法. 实际上,ORACLE在解析的过程中, 会将* 依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间. 8 减少访问数据库的次数 当执行每条SQL语句时, ORACLE在内部执行了许多工作: 解析SQL语句, 估算索引的利用率, 绑定变量 , 读数据块等等. 由此可见, 减少访问数据库的次数 , 就能实际上减少ORACLE的工作量. 例如, 以下有三种方法可以检索出雇员号等于0342或0291的职员. 方法1 (最低效) SELECT EMP_NAME , SALARY , GRADE FROM EM
15、P WHERE EMP_NO = 342; SELECT EMP_NAME , SALARY , GRADE FROM EMP WHERE EMP_NO = 291; 方法2 (次低效) DECLARE CURSOR C1 (E_NO NUMBER) IS SELECT EMP_NAME,SALARY,GRADE FROM EMP WHERE EMP_NO = E_NO; BEGIN OPEN C1(342); FETCH C1 INTO ,.,. ; . OPEN C1(291); FETCH C1 INTO ,.,. ; CLOSE C1; END; 方法3 (高效) SELECT A.EMP_NAME , A.SALARY , A.GRADE, B.EMP_NAME , B.S