ORACLE数据库的SQL性能优化

合集下载

oracle数据库性能调优

oracle数据库性能调优

oracle数据库性能调优⼀:注意WHERE⼦句中的连接顺序:ORACLE采⽤⾃下⽽上的顺序解析WHERE⼦句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最⼤数量记录的条件必须写在WHERE⼦句的末尾.尤其是“主键ID=?”这样的条件。

⼆: SELECT⼦句中避免使⽤ ‘ * ‘:ORACLE在解析的过程中, 会将'*' 依次转换成所有的列名, 这个⼯作是通过查询数据字典完成的, 这意味着将耗费更多的时间。

简单地讲,语句执⾏的时间越短越好(尤其对于系统的终端⽤户来说)。

⽽对于查询语句,由于全表扫描读取的数据多,尤其是对于⼤型表不仅查询速度慢,⽽且对磁盘IO造成⼤的压⼒,通常都要避免,⽽避免的⽅式通常是使⽤索引Index。

三:使⽤索引的优势与代价。

优势:1)索引是表的⼀个概念部分,⽤来提⾼检索数据的效率,ORACLE使⽤了⼀个复杂的⾃平衡B-tree结构. 通常,通过索引查询数据⽐全表扫描要快. 当ORACLE找出执⾏查询和Update语句的最佳路径时, ORACLE优化器将使⽤索引. 同样在联结多个表时使⽤索引也可以提⾼效率. 2)另⼀个使⽤索引的好处是,它提供了主键(primary key)的唯⼀性验证.。

那些LONG或LONG RAW数据类型, 你可以索引⼏乎所有的列. 通常, 在⼤型表中使⽤索引特别有效. 当然,你也会发现, 在扫描⼩表时,使⽤索引同样能提⾼效率.代价:虽然使⽤索引能得到查询效率的提⾼,但是我们也必须注意到它的代价. 索引需要空间来存储,也需要定期维护, 每当有记录在表中增减或索引列被修改时, 索引本⾝也会被修改. 这意味着每条记录的INSERT , DELETE , UPDATE将为此多付出4 , 5 次的磁盘I/O . 因为索引需要额外的存储空间和处理,那些不必要的索引反⽽会使查询反应时间变慢.。

⽽且表越⼤,影响越严重。

使⽤索引需要注意的地⽅:1、避免在索引列上使⽤NOT , 我们要避免在索引列上使⽤NOT, NOT会产⽣在和在索引列上使⽤函数相同的影响. 当ORACLE”遇到”NOT,他就会停⽌使⽤索引转⽽执⾏全表扫描.2、避免在索引列上使⽤计算.WHERE⼦句中,如果索引列是函数的⼀部分.优化器将不使⽤索引⽽使⽤全表扫描.举例:代码如下:低效:SELECT … FROM DEPT WHERE SAL * 12 > 25000;⾼效:SELECT … FROM DEPT WHERE SAL > 25000/12;3、避免在索引列上使⽤IS NULL和IS NOT NULL避免在索引中使⽤任何可以为空的列,ORACLE性能上将⽆法使⽤该索引.对于单列索引,如果列包含空值,索引中将不存在此记录. 对于复合索引,如果每个列都为空,索引中同样不存在此记录. 如果⾄少有⼀个列不为空,则记录存在于索引中.举例: 如果唯⼀性索引建⽴在表的A列和B列上, 并且表中存在⼀条记录的A,B值为(123,null) , ORACLE将不接受下⼀条具有相同A,B值(123,null)的记录(插⼊). 然⽽如果所有的索引列都为空,ORACLE将认为整个键值为空⽽空不等于空. 因此你可以插⼊1000 条具有相同键值的记录,当然它们都是空! 因为空值不存在于索引列中,所以WHERE⼦句中对索引列进⾏空值⽐较将使ORACLE停⽤该索引.代码如下:低效:(索引失效) SELECT … FROM DEPARTMENT WHERE DEPT_CODE IS NOT NULL;⾼效:(索引有效) SELECT … FROM DEPARTMENT WHERE DEPT_CODE >=0;4、注意通配符%的影响使⽤通配符的情况下Oracle可能会停⽤该索引。

浅谈Oracle数据库SQL性能优化

浅谈Oracle数据库SQL性能优化

1引言 随着 软件 技术 的不 断发展 , 系统性 能越来 越重 要。 信息系统都 离不开 数据库应用, 而O r a c l e  ̄据库的应 用系统一般规模 比较大, 如 何优化O RAC L E 数据库 的性能就显得尤为重要 。 为 了保证O r a c l e  ̄ 据库运行在最佳 的性能状态下 , 在信息系统开发之前就应该考虑数 据库 的优化策 略。 优化策略一般包括服务器操作系统参数调整 、 数 据库参数调整 、 网络 性能调整 、 应用程序s Q 分析及设计等几个 方面 , 本文就如何优化s Q L 语句的方法 来实现对O RAC L E 数据库性 能的优化 。
时 间 最 少 的也 就 是 所 谓 成 本 最 低 的一 种 方 法 。
( 1 ) 索引优化 要尽可 能的使 用索引 , 减少磁盘 的I / 0 操作 。 ( 2 ) 连接手段 在进行查询连接 时优化器将所有连接 的方法全 来强迫选择最佳索引 。 部列举 出来 , 计算每一种连接的成本, 选择成本最低的一种 。 如连接 例如: s e l e c t * f r o m t b— — r p — — b i z s _ a p p — — f i n— — l i s t w h e r e a p p— — d a t e > 时用 到的数据 无法 获得 , 一般 系统会使 用平均密度作为依据 , 估算 y s d a t e 一1 a n d a p p — da t e <s ys da t e a n d e x c h — i d = 9 1 0 0 1 ’ 可能的命 中率 。 如, 一个存储过程或触 发器 中, 有表达 式的值在编译 s 时无法得到 , 优 化器 就只能使用 它的平均密度 来估 计命 中的记录 O RAC L E 选择 的是 e x c h _ i d 索引为先 , a p p _ d a t e 索引在后 , 数。 例如: D EC L ARE @v lu a e mo n e y 如果用RUL E 规则, 也只会选择 e x c h _ i d索引 , 表分析 后仍 不改变 S E L E C T n a me F RO M e mp l o y e e W HE R E s a l a r y =@v a l u e 选择 ( OR AC L E 对确 定条件的优先级权值 比非确定条件高 ) 只是 由于 ̄ a l u e 的值在执行 前不知道 , 它 只能使用其平均密度来估 解决方法 : 计这条命令将要命 中的记录数 。 a ) e x c h _ i d= ‘ 9 1 0 0 1 ’改 为 e x c h _ i d l l ”=‘ 9 1 0 0 1 ’ b ) s e l e c t后面使用强制索引条件 , 强制不让 使用e x c h _ i d 索引 ( 3 ) 其他 手段 如 , 数 据表 空间和索引表 空间的分 离 , 关系密切 的表之间的表空 间的分离 , 表 空间的物理分布 , 都可以提 高应用的 s e l ct/ e * + n o _ i n d e x ( a i d x _ r p — b i z s a p p f i n l i s t 一 5 ) / * f r o m t b —r p — bi z s _ a pp — in f _ l i s t a w he r e a p p— d a t e >s ys da t e —l a n d 性 能。 a p p _ d a t e <s ys d a t e a nd e xc h _ i d =’ 9 1 0 0 1 ’ 遵守这些原则就可 以优化排序操作 , 提 高s Q L 查询性 能。 2 ) 对于多表关联查询 , 需要通过观察执行计划和S QL 语句的关 3 Or a c l e 数据库S QL 查询优化的过程和方法 联条件 , 找出当前索引路径 , 分析最佳索引路径 , 通过屏 蔽等手段让 3 . 1 Or a c l e  ̄ l 据 库S QL 查询 语 句 处理过 程 下转第 1 9 6 页

Oracle SQL性能优化之探究

Oracle SQL性能优化之探究
s t m e ou ce , 1 m t he ys e r s r s ee t re ui m nt of he us o e . o q re e s t c t m rs
K y w r s Q () i i a i n:R O:C O;S A: e f c e t S L e o d S L 1 t m z t o 1 B B G f iin Q
1 O a 的SA I rc G  ̄I I e : 性能的关系
1 1 r c e G 介绍 . O a I 的S A
O a ;精炼又 高效率 的S L rc Q 语句对一个大型 网站或大
型 数 据 库 来 讲 是 很 有 用 的 ,S L Q 的冗 余 造 成 系 统 与 资 源 的 一用 和 时 间 的 增 加 。什 么 样 的 S L 算 是 高 效 的 S L I i Q才 Q I ? 如 何 用 最 少 的 资 源 、 最 少 的 时 间 完 成 同样 的 事 情 呢 ?本 文 首 先 介 绍 O a l 的体 系 架 构 ,然 后 分 析 S L r ce Q 的 执 行 流 程 ,接 着 从 0 a l 优 化 器 如 何 查 看 执 行 计 划 , rce 如 何 使 用 索 引 , 最 后 结 合 实 例 , 从 各 个 角 度 来 描 述 怎
A tr ct I th d a bs a n e at bas a e ppl cat on ห้องสมุดไป่ตู้o t r i ng nd i i , w o ef ni a ef Ci ntl ge th q r r u1 fi e Y t e ue y es t
a c r i g t s r S u r r q e t T i r i l d s r b s o t o t m z t e S L t t m n i h c o d n o u e ’ q e y e u s ? h S a t c e e c i e h w o p i i e h Q s a e e t w t m n s e t . T e e p r m n a r s l s s o h t S L o t m z t o t a a e u e t e c u i r o a ya pc s h x e i e t l e u t h w t a , O p i i a i n h t c n r d c h o c p e f

oracle优化方案

oracle优化方案

千里之行,始于足下。

oracle优化方案Oracle优化方案Oracle数据库是当今企业界最受欢迎的关系型数据库管理系统之一。

但是,随着数据量的不断增加和业务需求的不断增长,数据库的性能问题也会渐渐变得突出。

因此,对Oracle数据库进行优化是提高系统性能和运行效率的关键。

本文将介绍几个常见的Oracle数据库优化方案,挂念您更好地管理和优化您的数据库环境。

1. 索引优化索引是提高查询性能的关键。

可以通过以下几个方面对索引进行优化:(1)合理选择索引类型:依据查询的特点和数据分布选择合适的索引类型,如B-tree索引、位图索引等。

(2)避开过多的索引:过多的索引会增加数据插入、更新和删除的成本,并降低查询性能。

只保留必要的索引,可以有效提高性能。

(3)定期重建和重新组织索引:定期重建和重新组织索引可以提高索引的查询效率,削减碎片和冗余。

2. SQL优化SQL语句是Oracle数据库的核心,对SQL进行优化可以显著提高数据库的性能。

以下是一些SQL优化的建议:第1页/共3页锲而不舍,金石可镂。

(1)优化查询语句:避开使用不必要的子查询,尽量使用连接查询代替子查询,削减查询次数。

同时,避开使用全表扫描,可以通过创建合适的索引来提高查询效率。

(2)避开使用不必要的OR运算符:OR运算符的查询效率较低,应尽量避开使用。

可以通过使用UNION或UNION ALL运算符代替OR运算符来提高性能。

(3)避开使用ORDER BY和GROUP BY子句:ORDER BY和GROUP BY子句会造成排序和分组操作,对于大数据集来说是格外耗时的。

假如可能,可以考虑使用其他方式来实现相同的功能。

3. 系统资源优化合理配置和管理系统资源是确保数据库运行稳定和高效的重要因素。

以下是一些建议:(1)合理安排内存:依据系统和数据库的实际需求,合理安排内存资源。

调整SGA(System Global Area)区域的大小,确保适当的内存安排给缓冲池和共享池。

oracle sql 优化技巧

oracle sql 优化技巧

oracle sql 优化技巧(实用版3篇)目录(篇1)1.Oracle SQL 简介2.优化技巧2.1 减少访问数据库次数2.2 选择最有效率的表名顺序2.3 避免使用 SELECT2.4 利用 DECODE 函数2.5 设置 ARRAYSIZE 参数2.6 使用 TRUNCATE 替代 DELETE2.7 多使用 COMMIT 命令2.8 合理使用索引正文(篇1)Oracle SQL 是一款广泛应用于各类大、中、小微机环境的高效、可靠的关系数据库管理系统。

为了提高 Oracle SQL 的性能,本文将为您介绍一些优化技巧。

首先,减少访问数据库的次数是最基本的优化方法。

Oracle 在内部执行了许多工作,如解析 SQL 语句、估算索引的利用率、读数据块等,这些都会大量耗费 Oracle 数据库的运行。

因此,尽量减少访问数据库的次数,可以有效提高系统性能。

其次,选择最有效率的表名顺序也可以明显提升 Oracle 的性能。

Oracle 解析器是按照从右到左的顺序处理 FROM 子句中的表名,因此,合理安排表名顺序,可以减少解析时间,提高查询效率。

在执行 SELECT 子句时,应尽量避免使用,因为 Oracle 在解析的过程中,会将依次转换成列名,这是通过查询数据字典完成的,耗费时间较长。

DECODE 函数也是一个很好的优化工具,它可以避免重复扫描相同记录,或者重复连接相同的表,提高查询效率。

在 SQLPlus 和 SQLForms 以及 ProC 中,可以重新设置 ARRAYSIZE 参数。

该参数可以明显增加每次数据库访问时的检索数据量,从而提高系统性能。

建议将该参数设置为 200。

当需要删除数据时,尽量使用 TRUNCATE 语句替代 DELETE 语句。

执行 TRUNCATE 命令时,回滚段不会存放任何可被恢复的信息,所有数据不能被恢复。

因此,TRUNCATE 命令执行时间短,且资源消耗少。

在使用 Oracle 时,尽量多使用 COMMIT 命令。

Oracle数据库参数优化

Oracle数据库参数优化

千里之行,始于足下。

Oracle数据库参数优化Oracle数据库参数优化是指通过调整数据库的配置参数,提高数据库的性能和稳定性。

下面是一些常见的Oracle数据库参数优化技巧:1. SGA参数优化:- 调整sga_target参数以控制SGA的大小。

SGA包括数据库缓冲区、共享池、重做日志缓冲区等,适当调整SGA的大小可以减少IO操作,提高数据库性能。

- 调整db_cache_size参数以增大数据库缓冲区的大小,提高数据块的访问速度。

- 调整shared_pool_size参数以增大共享池的大小,提高SQL语句的解析和执行效率。

2. PGA参数优化:- 调整pga_aggregate_target参数以控制PGA的大小。

PGA是用于处理SQL查询和排序的内存区域,适当调整PGA的大小可以减少磁盘IO操作,提高查询和排序的性能。

3. Redo日志参数优化:- 调整log_buffer参数以增大重做日志缓冲区的大小,减少频繁的重做日志刷新操作,提高数据库的写入性能。

- 调整log_checkpoint_timeout参数以控制重做日志刷新的频率,避免过于频繁的刷新。

4. 并行处理参数优化:- 调整parallel_max_servers参数以增大并行处理的资源限制,提高并行查询和并行DML操作的性能。

第1页/共2页锲而不舍,金石可镂。

- 调整parallel_min_servers参数以设置最小的并行处理资源数,避免并行操作的启动延迟。

5. SQL优化:- 使用合适的索引和优化的SQL语句,优化查询的执行计划。

- 使用绑定变量而不是直接将参数传递到SQL语句中,避免SQL重解析,提高性能。

6. 服务器参数优化:- 调整processes参数以增加数据库的并发连接数。

- 调整sessions参数以控制数据库的最大会话数。

- 调整open_cursors参数以增大打开游标的数量,避免游标溢出。

以上是一些常见的Oracle数据库参数优化技巧,但具体的优化策略需要根据实际情况进行调整,可以参考Oracle官方文档和专业的DBA建议。

Oracle数据库中SQL语句的优化与分析

Oracle数据库中SQL语句的优化与分析

第 7步 :Rn te Sa ee t u h t tm n
第 8步 :F t h R w f a Q e y ec o so ur
第 9步 :C o e t e C r o l s h u s r
下面 来详 细分 析这 些步骤 : 第 1 :C e t u s r 步 r a e a C r o
执 行每 个 S L 句 ,Oa l Q语 r ce需要 实现 很多 步骤 。Oa l r ce用 来 执 行 语句 的这 些 步骤 的组 合 被 称之 为执 行 计 划 。执 行 计 划 是 S L 化 中最为 复杂 也是最 为关 键 的部分 ,只有 知道 了 Oa l 在 O优 rce 内部到 底是 如何执 行 该 S L 句后 ,我们才 能知 道优 化器 选择 的 O语 执 行计 划是 否为 最优 的 。如何 分析 执行计 划 ,从 而找 出影 响性 能
s l ce ss ia l rno . e e td a u tbeo t
Ke wo d :QLOrceOpi zt n y r sS ; a l; t ai mi o
一ቤተ መጻሕፍቲ ባይዱ

引 言
第 6步 : P r l e i e t e S a e e t a a l l z h t t m n
S ,ny k o h w a l itmal x c t QL sae n, a o f m h ttee euinpa eeteo t z t n QL o l n w o Orce ne l e eueS ttme t cn cn r ta h x c t l wh r h pi ai r y we i o n mi o
的主要 问题 。下面 先 从分析 S L 句执 行步 骤 开始介 绍 ,再介 绍 O 语 如 何分 析执行 计划 。优 化器 有 时也被 称 为查询 优 化器 ,这 是因 为 查 询是 影响数 据库 性 能最主 要 的部分 ,优 化器 是所 有 关系数 据库 引擎 中的最神 秘 、最 富挑 战性 的部件 之一 ,从 性 能的角 度看 也是 最 重要 的部分 ,它 性 能的高低 直接 关 系到数 据库 性能 的好坏 。 二 、0 a l r c e的优化规 则 ( )什 么是优 化 一

Oracle数据库性能优化分析

Oracle数据库性能优化分析

千里之行,始于足下。

Oracle数据库性能优化分析Oracle数据库性能优化分析是指对Oracle数据库进行综合性能分析和优化的过程。

通过分析数据库的运行状况、识别潜在的性能瓶颈、确定解决方案并实施优化措施,可以提高数据库的性能和效率。

以下是Oracle数据库性能优化分析的一般步骤:1. 收集性能数据:通过Oracle的性能监控工具,如AWR报告、统计信息收集等,收集数据库的性能数据,包括CPU利用率、I/O响应时间、锁定情况等。

2. 确定性能瓶颈:通过分析性能数据,确定数据库中存在的性能瓶颈,如高CPU使用率、高IO等待、长时间的锁等待等。

3. 优化SQL语句:分析执行频次较高的SQL语句,通过重写SQL语句、调整索引和统计信息等方式,优化SQL语句的执行计划,减少IO开销和CPU消耗。

4. 优化数据库结构:根据应用的需求和查询模式,调整表结构、分区策略、索引设计等,以提高查询性能和数据访问效率。

5. 优化数据库配置参数:调整数据库的配置参数,包括缓冲区大小、日志大小、并发连接数等,以最大限度地利用硬件资源,提高数据库的吞吐量和响应时间。

6. 确保数据完整性和一致性:通过使用合适的约束和触发器,确保数据的完整性和一致性,防止数据错误和冲突对性能造成负面影响。

第1页/共2页锲而不舍,金石可镂。

7. 监控和调优:定期监控数据库的性能指标,如响应时间、吞吐量等,及时识别和解决潜在的性能问题,保持数据库的高可用性和性能稳定性。

需要注意的是,性能优化是一个综合性的工作,需要结合具体的应用场景和需求来进行分析和优化,没有一种通用的解决方案,需要根据实际情况进行定制化的优化措施。

同时,性能优化是一个持续改进的过程,需要定期评估数据库的性能状况,并根据需求进行调整和优化。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

ORACLE数据库的SQL性能优化系列知识(共十四讲)原文作者: black_snail关键字ORA CEL SQL Performance tuning出处ORACLE SQL性能优化系列(一)1. 选用适合的ORACLE优化器ORACLE的优化器共有3种:a. RULE (基于规则)b. COST (基于成本)c. CHOOSE (选择性)设置缺省的优化器,可以通过对init.ora文件中OPTIMIZER_MODE参数的各种声明,如RULE,COST,CHOOSE,A LL_ROWS,FIRST_ROWS,你当然也在SQL语句级或是会话(session)级对其进行覆盖.为了使用基于成本的优化器(CBO, Cost-Based Optimizer) , 你必须经常运行analyze 命令,以增加数据库中的对象统计信息(object statistics)的准确性.如果数据库的优化器模式设置为选择性(CHOOSE),那么实际的优化器模式将和是否运行过analyze命令有关. 如果table已经被analyze过, 优化器模式将自动成为CBO , 反之,数据库将采用RULE形式的优化器.在缺省情况下,ORA CLE采用CHOOSE优化器, 为了避免那些不必要的全表扫描(full table scan) , 你必须尽量避免使用CHOOSE优化器,而直接采用基于规则或者基于成本的优化器.2. 访问Table的方式ORACLE 采用两种访问表中记录的方式:a. 全表扫描全表扫描就是顺序地访问表中每条记录. ORACLE采用一次读入多个数据块(database block)的方式优化全表扫描.b. 通过ROWID访问表你可以采用基于ROWID的访问方式情况,提高访问表的效率, ROWID包含了表中记录的物理位置信息.ORACLE采用索引(INDEX)实现了数据和存放数据的物理位置(ROWID)之间的联系.通常索引提供了快速访问ROWID的方法,因此那些基于索引列的查询就可以得到性能上的提高.3. 共享SQL语句为了不重复解析相同的SQL语句,在第一次解析之后, ORA CLE将SQL语句存放在内存中.这块位于系统全局区域SGA(system global area)的共享池(shared buffer pool)中的内存可以被所有的数据库用户共享. 因此,当你执行一个SQL语句(有时被称为一个游标)时,如果它和之前的执行过的语句完全相同, ORACLE就能很快获得已经被解析的语句以及最好的执行路径. ORACLE的这个功能大大地提高了SQL的执行性能并节省了内存的使用。

可惜的是ORA CLE只对简单的表提供高速缓冲(cache buffering),这个功能并不适用于多表连接查询。

数据库管理员必须在init.ora中为这个区域设置合适的参数,当这个内存区域越大,就可以保留更多的语句,当然被共享的可能性也就越大了.当你向ORACLE提交一个SQL语句,ORA CLE会首先在这块内存中查找相同的语句.这里需要注明的是,ORACLE对两者采取的是一种严格匹配,要达成共享,SQL语句必须完全相同(包括空格,换行等).共享的语句必须满足三个条件:A. 字符级的比较:当前被执行的语句和共享池中的语句必须完全相同. 例如:SELECT * FROM EMP;和下列每一个都不同SELECT * from EMP;Select * From Emp;SELECT * FROM EMP;B. 两个语句所指的对象必须完全相同: 例如:用户对象名如何访问Jack sal_limit private synonymWork_city public synonymPlant_detail public synonymJill sal_limit private synonymWork_city public synonymPlant_detail table owner考虑一下下列SQL语句能否在这两个用户之间共享.SQL能否共享原因select max(sal_cap) from sal_limit;不能,每个用户都有一个private synonym - sal_limit , 它们是不同的对象select count(*) from work_city where sdesc like 'NEW%';能,两个用户访问相同的对象public synonym - work_cityselect a.sdesc,b.location from work_city a , plant_detail b where a.city_id = b.city_id不能,用户jack 通过private synonym访问plant_detail 而jill 是表的所有者,对象不同.C. 两个SQL语句中必须使用相同的名字的绑定变量(bind variables)例如:第一组的两个SQL语句是相同的(可以共享),而第二组中的两个语句是不同的(即使在运行时,赋于不同的绑定变量相同的值)a.select pin , name from people where pin = :blk1.pin;select pin , name from people where pin = :blk1.pin;b.select pin , name from people where pin = :blk1.ot_ind;select pin , name from people where pin = :blk1.ov_ind;ORACLE SQL性能优化系列(二)4. 选择最有效率的表名顺序(只在基于规则的优化器中有效)ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,因此FROM子句中写在最后的表(基础表driving table)将被最先处理. 在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表.当ORACLE处理多个表时, 会运用排序及合并的方式连接它们.首先,扫描第一个表(FROM子句中最后的那个表)并对记录进行派序,然后扫描第二个表(FROM子句中从最后算起的第二个表),最后将所有从第二个表中检索出的记录与第一个表中合适记录进行合并.例如:表TA B1 16,384 条记录表TA B2 1 条记录选择TAB2作为基础表(最好的方法)select count(*) from tab1,tab2 执行时间0.96秒选择TAB2作为基础表(不佳的方法)select count(*) from tab2,tab1 执行时间26.09秒如果有3个以上的表连接查询, 那就需要选择交叉表(intersection table)作为基础表, 交叉表是指那个被其他表所引用的表.例如:EMP表描述了LOCATION表和CATEGORY表的交集.SELECT *FROM LOCATION L ,CATEGORY C,EMP EWHERE E.EMP_NO BETW EEN 1000 AND 2000AND E.CAT_NO = C.CAT_NOAND E.LOCN = L.LOCN将比下列SQL更有效率SELECT *FROM EMP E ,LOCATION L ,CATEGORY CWHERE E.CAT_NO = C.CAT_NOAND E.LOCN = L.LOCNAND E.EMP_NO BETWEEN 1000 A ND 20005. W HERE子句中的连接顺序.ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE 条件之前,那些可以过滤掉最大数量记录的条件必须写在W HERE子句的末尾.例如:(低效,执行时间156.3秒)SELECT …FROM EMP EWHERE SA L > 50000AND JOB = …MANA GER'AND 25 < (SELECT COUNT(*) FROM EMP WHERE MGR=E.EMPNO);(高效,执行时间10.6秒)SELECT …FROM EMP EWHERE 25 < (SELECT COUNT(*) FROM EMP WHERE MGR=E.EMPNO)AND SAL > 50000AND JOB = …MANA GER';6. SELECT子句中避免使用“*”当你想在SELECT子句中列出所有的COLUMN时,使用动态SQL列引用“*”是一个方便的方法.不幸的是,这是一个非常低效的方法.实际上,ORA CLE在解析的过程中, 会将“*”依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间.7. 减少访问数据库的次数当执行每条SQL语句时,ORACLE在内部执行了许多工作:解析SQL语句,估算索引的利用率, 绑定变量,读数据块等等.由此可见,减少访问数据库的次数,就能实际上减少ORA CLE的工作量.例如, 以下有三种方法可以检索出雇员号等于0342或0291的职员.方法1 (最低效)SELECT EMP_NAME , SA LARY , GRA DEFROM EMPWHERE EMP_NO = 342;SELECT EMP_NAME , SA LARY , GRA DEFROM EMPWHERE EMP_NO = 291;方法2 (次低效)DECLARECURSOR C1 (E_NO NUMBER) ISSELECT EMP_NAME,SA LA RY,GRADEFROM EMPWHERE EMP_NO = E_NO;BEGINOPEN C1(342);FETCH C1 INTO …,..,.. ;…..OPEN C1(291);FETCH C1 INTO …,..,.. ;CLOSE C1;END;方法3 (高效)SELECT A.EMP_NAME , A.SA LARY , A.GRA DE,B.EMP_NAME , B.SA LA RY , B.GRADEFROM EMP A,EMP BWHERE A.EMP_NO = 342AND B.EMP_NO = 291;注意: 在SQL*Plus , SQL*Forms和Pro*C中重新设置A RRA YSIZE参数, 可以增加每次数据库访问的检索数据量,建议值为200ORACLE SQL性能优化系列(三)8. 使用DECODE函数来减少处理时间使用DECODE函数可以避免重复扫描相同记录或重复连接相同的表.例如:SELECT COUNT(*),SUM(SA L)FROM EMPWHERE DEPT_NO = 0020AND ENAME LIKE …SMITH%‟;SELECT COUNT(*),SUM(SA L)FROM EMPWHERE DEPT_NO = 0030AND ENAME LIKE …SMITH%‟;你可以用DECODE函数高效地得到相同结果SELECT COUNT(DECODE(DEPT_NO,0020,'X',NULL)) D0020_COUNT,COUNT(DECODE(DEPT_NO,0030,'X',NULL)) D0030_COUNT,SUM(DECODE(DEPT_NO,0020,SA L,NULL)) D0020_SA L,SUM(DECODE(DEPT_NO,0030,SA L,NULL)) D0030_SA LFROM EMP WHERE ENAME LIKE …SMITH%';类似的,DECODE函数也可以运用于GROUP BY 和ORDER BY子句中.9. 整合简单,无关联的数据库访问如果你有几个简单的数据库查询语句,你可以把它们整合到一个查询中(即使它们之间没有关系) 例如:SELECT NAMEFROM EMPWHERE EMP_NO = 1234;SELECT NAMEFROM DPTWHERE DPT_NO = 10 ;SELECT NAMEFROM CATWHERE CAT_TYPE = …RD';上面的3个查询可以被合并成一个:SELECT , , FROM CAT C , DPT D , EMP E,DUA L XWHERE NVL(…X',X.DUMMY) = NVL(…X',E.ROWID(+))AND NVL(…X',X.DUMMY) = NVL(…X',D.ROWID(+))AND NVL(…X',X.DUMMY) = NVL(…X',C.ROWID(+))AND E.EMP_NO(+) = 1234AND D.DEPT_NO(+) = 10AND C.CAT_TYPE(+) = …RD';(译者按: 虽然采取这种方法,效率得到提高,但是程序的可读性大大降低,所以读者还是要权衡之间的利弊)10. 删除重复记录最高效的删除重复记录方法(因为使用了ROWID)DELETE FROM EMP EWHERE E.ROWID > (SELECT MIN(X.ROWID)FROM EMP X WHERE X.EMP_NO = E.EMP_NO);11. 用TRUNCATE替代DELETE当删除表中的记录时,在通常情况下,回滚段(rollback segments )用来存放可以被恢复的信息. 如果你没有COMMIT事务,ORA CLE会将数据恢复到删除之前的状态(准确地说是恢复到执行删除命令之前的状况)而当运用TRUNCATE时, 回滚段不再存放任何可被恢复的信息.当命令运行后,数据不能被恢复.因此很少的资源被调用,执行时间也会很短.(译者按: TRUNCATE只在删除全表适用,TRUNCATE是DDL不是DML)12. 尽量多使用COMMIT只要有可能,在程序中尽量多使用COMMIT, 这样程序的性能得到提高,需求也会因为COMMIT所释放的资源而减少:COMMIT所释放的资源:a. 回滚段上用于恢复数据的信息.b. 被程序语句获得的锁c. redo log buffer 中的空间d. ORACLE为管理上述3种资源中的内部花费(译者按: 在使用COMMIT时必须要注意到事务的完整性,现实中效率和事务完整性往往是鱼和熊掌不可得兼)ORACLE SQL性能优化系列(四)13. 计算记录条数和一般的观点相反, count(*) 比count(1)稍快, 当然如果可以通过索引检索,对索引列的计数仍旧是最快的. 例如COUNT(EMPNO)(译者按: 在CSDN论坛中,曾经对此有过相当热烈的讨论,作者的观点并不十分准确,通过实际的测试,上述三种方法并没有显著的性能差别)14. 用Where子句替换HA VING子句避免使用HA VING子句, HA VING 只会在检索出所有记录之后才对结果集进行过滤. 这个处理需要排序,总计等操作.如果能通过WHERE子句限制记录的数目,那就能减少这方面的开销. 例如:低效:SELECT REGION,A VG(LOG_SIZE)FROM LOCATIONGROUP BY REGIONHA VING REGION REGION != …SYDNEY'AND REGION != …PERTH'高效:SELECT REGION,A VG(LOG_SIZE)FROM LOCATIONWHERE REGION REGION != …SYDNEY'AND REGION != …PERTH'GROUP BY REGION(译者按: HA VING 中的条件一般用于对一些集合函数的比较,如COUNT() 等等. 除此而外,一般的条件应该写在WHERE子句中)15. 减少对表的查询在含有子查询的SQL语句中,要特别注意减少对表的查询. 例如:低效SELECT TA B_NAMEFROM TA BLESWHERE TAB_NAME = ( SELECT TAB_NAMEFROM TA B_COLUMNSWHERE VERSION = 604)AND DB_VER= ( SELECT DB_VERFROM TA B_COLUMNSWHERE VERSION = 604)高效SELECT TA B_NAMEFROM TA BLESWHERE (TA B_NAME,DB_VER)= ( SELECT TA B_NAME,DB_VER)FROM TA B_COLUMNSWHERE VERSION = 604)Update 多个Column 例子:低效:UPDATE EMPSET EMP_CAT = (SELECT MAX(CATEGORY) FROM EMP_CATEGORIES),SAL_RANGE = (SELECT MAX(SAL_RANGE) FROM EMP_CATEGORIES)WHERE EMP_DEPT = 0020;高效:UPDATE EMPSET (EMP_CAT, SA L_RANGE)= (SELECT MAX(CATEGORY) , MAX(SA L_RANGE)FROM EMP_CATEGORIES)WHERE EMP_DEPT = 0020;16. 通过内部函数提高SQL效率.SELECT H.EMPNO,E.ENAME,H.HIST_TYPE,T.TYPE_DESC,COUNT(*)FROM HISTORY_TYPE T,EMP E,EMP_HISTORY HWHERE H.EMPNO = E.EMPNOAND H.HIST_TYPE = T.HIST_TYPEGROUP BY H.EMPNO,E.ENAME,H.HIST_TYPE,T.TYPE_DESC;通过调用下面的函数可以提高效率.FUNCTION LOOKUP_HIST_TYPE(TYP IN NUMBER) RETURN VA RCHAR2ASTDESC VA RCHA R2(30);CURSOR C1 ISSELECT TYPE_DESCFROM HISTORY_TYPEWHERE HIST_TYPE = TYP;BEGINOPEN C1;FETCH C1 INTO TDESC;CLOSE C1;RETURN (NVL(TDESC,'?'));END;FUNCTION LOOKUP_EMP(EMP IN NUMBER) RETURN VA RCHAR2ASENAME VA RCHA R2(30);CURSOR C1 ISSELECT ENAMEFROM EMPWHERE EMPNO=EMP;BEGINOPEN C1;FETCH C1 INTO ENAME;CLOSE C1;RETURN (NVL(ENAME,'?'));END;SELECT H.EMPNO,LOOKUP_EMP(H.EMPNO),H.HIST_TYPE,LOOKUP_HIST_TYPE(H.HIST_TYPE),COUNT(*)FROM EMP_HISTORY HGROUP BY H.EMPNO , H.HIST_TYPE;译者按: 经常在论坛中看到如'能不能用一个SQL写出….' 的贴子, 殊不知复杂的SQL往往牺牲了执行效率. 能够掌握上面的运用函数解决问题的方法在实际工作中是非常有意义的ORACLE SQL性能优化系列(五)17. 使用表的别名(Alias)当在SQL语句中连接多个表时, 请使用表的别名并把别名前缀于每个Column上.这样一来,就可以减少解析的时间并减少那些由Column歧义引起的语法错误.译者注: Column歧义指的是由于SQL中不同的表具有相同的Column名,当SQL语句中出现这个Column时,SQL解析器无法判断这个Column的归属18. 用EXISTS替代IN在许多基于基础表的查询中,为了满足一个条件,往往需要对另一个表进行联接.在这种情况下, 使用EXISTS(或NOT EXISTS)通常将提高查询的效率.低效:SELECT *FROM EMP (基础表)WHERE EMPNO > 0AND DEPTNO IN (SELECT DEPTNOFROM DEPTWHERE LOC = …MELB')高效:SELECT *FROM EMP (基础表)WHERE EMPNO > 0AND EXISTS (SELECT …X'FROM DEPTWHERE DEPT.DEPTNO = EMP.DEPTNOAND LOC = …MELB')译者按: 相对来说,用NOT EXISTS替换NOT IN 将更显著地提高效率,下一节中将指出19. 用NOT EXISTS替代NOT IN在子查询中,NOT IN子句将执行一个内部的排序和合并. 无论在哪种情况下,NOT IN都是最低效的(因为它对子查询中的表执行了一个全表遍历). 为了避免使用NOT IN ,我们可以把它改写成外连接(Outer Joins)或NOT EXISTS.例如:SELECT …FROM EMPWHERE DEPT_NO NOT IN (SELECT DEPT_NOFROM DEPTWHERE DEPT_CAT='A');为了提高效率.改写为:(方法一: 高效)SELECT ….FROM EMP A,DEPT BWHERE A.DEPT_NO = B.DEPT(+)AND B.DEPT_NO IS NULLAND B.DEPT_CAT(+) = …A'(方法二: 最高效)SELECT ….FROM EMP EWHERE NOT EXISTS (SELECT …X'FROM DEPT DWHERE D.DEPT_NO = E.DEPT_NOAND DEPT_CAT = …A');ORACLE SQL性能优化系列(六)20. 用表连接替换EXISTS通常来说, 采用表连接的方式比EXISTS更有效率SELECT ENAMEFROM EMP EWHERE EXISTS (SELECT …X'FROM DEPTWHERE DEPT_NO = E.DEPT_NOAND DEPT_CAT = …A');(更高效)SELECT ENAMEFROM DEPT D,EMP EWHERE E.DEPT_NO = D.DEPT_NOAND DEPT_CAT = …A' ;译者按: 在RBO的情况下,前者的执行路径包括FILTER,后者使用NESTED LOOP21. 用EXISTS替换DISTINCT当提交一个包含一对多表信息(比如部门表和雇员表)的查询时,避免在SELECT子句中使用DISTINCT. 一般可以考虑用EXIST替换例如:低效:SELECT DISTINCT DEPT_NO,DEPT_NAMEFROM DEPT D,EMP EWHERE D.DEPT_NO = E.DEPT_NO高效:SELECT DEPT_NO,DEPT_NAMEFROM DEPT DWHERE EXISTS ( SELECT …X'FROM EMP EWHERE E.DEPT_NO = D.DEPT_NO);EXISTS 使查询更为迅速,因为RDBMS核心模块将在子查询的条件一旦满足后,立刻返回结果.22. 识别'低效执行'的SQL语句用下列SQL工具找出低效SQL:SELECT EXECUTIONS , DISK_READS, BUFFER_GETS,ROUND((BUFFER_GETS-DISK_READS)/BUFFER_GETS,2) Hit_radio,ROUND(DISK_READS/EXECUTIONS,2) Reads_per_run,SQL_TEXTFROM V$SQLAREAWHERE EXECUTIONS>0AND BUFFER_GETS > 0AND (BUFFER_GETS-DISK_REA DS)/BUFFER_GETS < 0.8ORDER BY 4 DESC;译者按: 虽然目前各种关于SQL优化的图形化工具层出不穷,但是写出自己的SQL工具来解决问题始终是一个最好的方法23. 使用TKPROF工具来查询SQL性能状态SQL trace 工具收集正在执行的SQL的性能状态数据并记录到一个跟踪文件中. 这个跟踪文件提供了许多有用的信息,例如解析次数.执行次数,CPU使用时间等.这些数据将可以用来优化你的系统.设置SQL TRA CE在会话级别: 有效ALTER SESSION SET SQL_TRACE TRUE设置SQL TRACE 在整个数据库有效仿, 你必须将SQL_TRA CE参数在init.ora中设为TRUE, USER_DUMP_DEST参数说明了生成跟踪文件的目录译者按: 这一节中,作者并没有提到TKPROF的用法, 对SQL TRA CE的用法也不够准确, 设置SQL TRA CE首先要在init.ora中设定TIMED_STATISTICS, 这样才能得到那些重要的时间状态. 生成的trace文件是不可读的,所以要用TKPROF工具对其进行转换,TKPROF有许多执行参数. 大家可以参考ORACLE手册来了解具体的配置.ORACLE SQL性能优化系列(七)24. 用EXPLAIN PLA N 分析SQL语句EXPLAIN PLAN 是一个很好的分析SQL语句的工具,它甚至可以在不执行SQL的情况下分析语句. 通过分析,我们就可以知道ORACLE是怎么样连接表,使用什么方式扫描表(索引扫描或全表扫描)以及使用到的索引名称.你需要按照从里到外,从上到下的次序解读分析的结果. EXPLAIN PLAN分析的结果是用缩进的格式排列的, 最内部的操作将被最先解读, 如果两个操作处于同一层中,带有最小操作号的将被首先执行.NESTED LOOP是少数不按照上述规则处理的操作, 正确的执行路径是检查对NESTED LOOP提供数据的操作,其中操作号最小的将被最先处理.译者按: 通过实践, 感到还是用SQLPLUS中的SET TRA CE 功能比较方便.Trace功能的设置:SQL> connect sys/XXXXXXX已连接。

相关文档
最新文档