数据库SQL性能优化
1.in与exists的效率比较
in是把外表和内表作hash 连接,而exists 是对外表作loop 循环,每次loop 循环再对内表进行查询。一直以来认为exists 比in 效率高的说法是不准确的。如果查询的两个表大小相当,那么用in 和exists 差别不大。
如果两个表中一个较小,一个是大表,则子查询表大的用exists,子查询表小的用in:
例如:
表A(小表),表B(大表)1:
select * from A where cc in (select cc from B)效率低,用到了A 表上cc 列的索引;
select * from A where exists(select cc from B where cc=A.cc)效率高,用到了B 表上cc 列的索引。
2.not in 和not exists
如果查询语句使用了not in 那么内外表都进行全表扫描,没有用到索引;
而not extsts 的子查询依然能用到表上的索引。所以无论那个表大,用not exists 都比not in 要快。
3.is null 和is not null的比较
索引列是不允许有null值的,也就是说如果某列存在空值,即使对该列建索引也不会提高性能。推荐方案:用其它相同功能的操作运算代替,如:a is not null 改为 a>0 或a>’ ’等。不允许字段为空,而用一个缺省值代替空值,如申请中状态字段不允许为空,缺省为申请。
4.LIKE操作符
用YW_YHJBQK表中营业编号后面的户标识号可来查询营业编号 YY_BH LIKE ‘%5400%’ 这个条件会产生全表扫描,这种查询不会引用索引,如果改成YY_BH LIKE ’X5400%’ OR YY_BH LIKE ’B5400%’ 则会利用YY_BH的索引进行两个范围的查询,性能肯定大大提高。
5.union操作符
union在进行表链接后会筛选掉重复的记录,所以在表链接后会对所产生的结果集进行排序运算,删除重复的记录再返回结果。实际大部分应用中是不会产生重复的记录:最常见的是过程表与历史表UNION。如:
select
*
from
gc_dfys
union
select
*
from
ls_jg_dfys
select
*
from
employss
where
first_name||
''
||last_name =
'Beill Cliton'
;这样的话系统优化器对基于last_name创建的索引没有使用。
Select
*
from
zl_yhjbqk
where
dy_dj =
'1KV以下'
and
xh_bz=1
Select
*
from
zl_yhjbqk
where
xh_bz=1
and
dy_dj =
'1KV以下'
substr(hbs_bh,1,4)=’5400’,优化处理:hbs_bh
like
‘5400%’
trunc(sk_rq)=trunc(sysdate), 优化处理:sk_rq>=trunc(sysdate)
and
sk_rq<trunc(sysdate+1)
‘X’ || hbs_bh>’X5400021452’,优化处理:hbs_bh>’5400021542’
sk_rq+5=sysdate,优化处理:sk_rq=sysdate-5
ys_df>cx_df,无法进行优化
qc_bh || kh_bh=’5400250000’,优化处理:qc_bh=’5400’
and
kh_bh=’250000’
ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾.
12. SELECT子句中避免使用 ‘ * ‘:ORACLE在解析的过程中, 会将’*’ 依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间。
13.减少访问数据库的次数:ORACLE在内部执行了许多工作: 解析SQL语句, 估算索引的利用率, 绑定变量 , 读数据块等。
14.在SQL*Plus , SQL*Forms和Pro*C中重新设置ARRAYSIZE参数, 可以增加每次数据库访问的检索数据量 ,建议值为200。
15. 整合简单,无关联的数据库访问:如果你有几个简单的数据库查询语句,你可以把它们整合到一个查询中(即使它们之间没有关系) 。
16. 删除重复记录:最高效的删除重复记录方法 ( 因为使用了ROWID)例子:DELETE
FROM
EMP E
WHERE
E.ROWID > (
SELECT
MIN
(X.ROWID)
FROM
EMP X
WHERE
X.EMP_NO = E.EMP_NO)。
17. 用TRUNCATE替代DELETE:
18. 尽量多使用COMMIT
COMMIT所释放的资源:
a. 回滚段上用于恢复数据的信息.
b. 被程序语句获得的锁
c. redo log buffer 中的空间
19.用where替换having子句,避免使用HAVING子句, HAVING 只会在检索出所有记录之后才对结果集进行过滤. 这个处理需要排序,总计等操作.如果能通过WHERE子句限制记录的数目,那就能减少这方面的开销. (非oracle中)on、where、having这三个都可以加条件的子句中,on是最先执行where次之,having最后.因为on是先把不符合条件的记录过滤后才进行统计,它就可以减少中间运算要处理的数据,按理说应该速度是最快的,where也应该比having快点的,因为它过滤数据后才进行sum,在两个表联接时才用on的,所以在一个表的时候,就剩下where跟having比较了.在这单表查询统计的情况下,如果要过滤的条件没有涉及到要计算字段,那它们的结果是一样的,只是where可以使用rushmore技术,而having就不能,在速度上后者要慢如果要涉及到计算的字 段,就表示在没计算之前,这个字段的值是不确定的,根据上篇写的工作流程,where的作用时间是在计算之前就完成的,而having就是在计算后才起作 用的,所以在这种情况下,两者的结果会不同。在多表联接查询时,on比where更早起作用。系统首先根据各个表之间的联接条件,把多个表合成一个临时表 后,再由where进行过滤,然后再计算,计算完后再由having进行过滤。由此可见,要想过滤条件起到正确的作用,首先要明白这个条件应该在什么时候起作用,然后再决定放在那里。
20.使用表的别名(Alias)当在SQL语句中连接多个表时, 请使用表的别名并把别名前缀于每个Column上.这样一来,就可以减少解析的时间并减少那些由Column歧义引起的语法错误。
21. 用索引提高效率:
低效: (索引不被使用)
SELECT
DEPT_CODE
FROM
DEPT
ORDER
BY
DEPT_TYPE
高效: (使用索引)
SELECT
DEPT_CODE
FROM
DEPT
WHERE
DEPT_TYPE > 0
低效:
SELECT
JOB ,
AVG
(SAL)
FROM
EMP
GROUP
by
JOB
HAVING
JOB = ‘PRESIDENT
'
OR JOB = ‘MANAGER'
高效:
SELECT
JOB ,
AVG
(SAL)
FROM
EMP
WHERE
JOB = ‘PRESIDENT
'
OR JOB = ‘MANAGER'
GROUP
by
JOB