ORACLE优化器之RBO与CBO

转载自:潇湘隐者 http://www.cnblogs.com/kerrycode/p/3842215.html
基础概念 Oracle发展历程:5->6->7->8->8i->9i->10g->11g->12c
**Oracle数据库中的优化器又叫查询优化器(Query Optimizer)。它是SQL分析和执行的优化工具,它负责生成、制定SQL的执行计划。**Oracle的优化器有两种,基于规则的优化器(RBO)与基于代价的优化器(CBO)。
RBO: Rule-Based Optimization 基于规则的优化器
CBO: Cost-Based Optimization 基于代价的优化器
RBO自ORACLE 6以来被采用,一直沿用至ORACLE 9i。ORACLE 10g开始,ORACLE已经彻底丢弃了RBO,它有着一套严格的使用规则,只要你按照它去写SQL语句,无论数据表中的内容怎样,也不会影响到你的“执行计划”,也就是说RBO对数据不“敏感”;它根据ORACLE指定的优先顺序规则,对指定的表进行执行计划的选择。比如在规则中,索引的优先级大于全表扫描; RBO是根据可用的访问路径以及访问路径等级来选择执行计划,在RBO中,SQL的写法往往会影响执行计划,它要求开发人员非常了解RBO的各项细则,菜鸟写出来的SQL脚本性能可能非常差。
CBO是一种比RBO更加合理、可靠的优化器,它是从ORACLE 8中开始引入,但到ORACLE 9i 中才逐渐成熟,在ORACLE 10g中完全取代RBO, CBO是计算各种可能“执行计划”的“代价”,即COST,从中选用COST最低的执行方案,作为实际运行方案。它依赖数据库对象的统计信息,统计信息的准确与否会影响CBO做出最优的选择。如果对一次执行SQL时发现涉及对象(表、索引等)没有被分析、统计过,那么ORACLE会采用一种叫做动态采样的技术,动态的收集表和索引上的一些数据信息。
关于RBO与CBO,我有个形象的比喻:大数据时代到来以前,做生意或许凭借多年累计下来的经验(RBO)就能够很好的做出决策,跟随市场变化。但是大数据时代,如果做生意还是靠以前凭经验做决策,而不是靠大数据、数据分析、数据挖掘做决策,那么就有可能做出错误的决策。这也就是越来越多的公司对BI、数据挖掘越来越重视的缘故,像电商、游戏、电信等行业都已经大规模的应用。
CBO与RBO的优劣 CBO优于RBO是因为RBO是一种呆板、过时的优化器,它只认规则,对数据不敏感。毕竟规则是死的,数据是变化的,这样生成的执行计划往往是不可靠的,不是最优的,CBO由于RBO可以从很多方面体现。下面请看一个例子,此案例来自于《让Oracle跑得更快》。

SQL> create table test as select 1 id ,object_name from dba_objects; Table created.SQL> create index idx_test on test(id); Index created.SQL> update test set id=100 where rownum =1; 1 row updated.SQL> select id, count(1) from test group by id; IDCOUNT(1) ---------- ---------- 1001 150314

SQL> select /*+ rule */ * from test where id =100; Execution Plan ---------------------------------------------------------- Plan hash value: 2473784974------------------------------------------------ | Id| Operation| Name| ------------------------------------------------ |0 | SELECT STATEMENT|| |1 |TABLE ACCESS BY INDEX ROWID| TEST| |*2 |INDEX RANGE SCAN| IDX_TEST | ------------------------------------------------ Predicate Information (identified by operation id): --------------------------------------------------- 2 - access("ID"=100)Note ----- - rule based optimizer used (consider using cbo)Statistics ---------------------------------------------------------- 1recursive calls 0db block gets 3consistent gets 0physical reads 0redo size 588bytes sent via SQL*Net to client 469bytes received via SQL*Net from client 2SQL*Net roundtrips to/from client 0sorts (memory) 0sorts (disk) 1rows processed

SQL> select /*+ rule */ * from test where id=1; 50314 rows selected.Execution Plan ---------------------------------------------------------- Plan hash value: 2473784974------------------------------------------------ | Id| Operation| Name| ------------------------------------------------ |0 | SELECT STATEMENT|| |1 |TABLE ACCESS BY INDEX ROWID| TEST| |*2 |INDEX RANGE SCAN| IDX_TEST | ------------------------------------------------ Predicate Information (identified by operation id): --------------------------------------------------- 2 - access("ID"=1)Note ----- - rule based optimizer used (consider using cbo)Statistics ---------------------------------------------------------- 1recursive calls 0db block gets 7012consistent gets 97physical reads 0redo size 2243353bytes sent via SQL*Net to client 37363bytes received via SQL*Net from client 3356SQL*Net roundtrips to/from client 0sorts (memory) 0sorts (disk) 50314rows processed

从执行计划可以看出,RBO的执行计划让人有点失望,对于ID=1,几乎所有的数据全部符合谓词条件,走索引只能增加额外的开销(因为ORACLE首先要访问索引数据块,在索引上找到了对应的键值,然后按照键值上的ROWID再去访问表中相应数据),既然我们几乎要访问所有表中的数据,那么全表扫描自然是最优的选择。而RBO选择了错误的执行计划。可以对比一下CBO下SQL的执行计划,显然它对数据敏感,执行计划及时的根据数据量做了调整,当查询条件为1时,它走全表扫描;当查询条件为100时,它走区间索引扫描。如下所示:
SQL> select * from test where id=1; 50314 rows selected.Execution Plan ---------------------------------------------------------- Plan hash value: 1357081020-------------------------------------------------------------------------- | Id| Operation| Name | Rows| Bytes | Cost (%CPU)| Time| -------------------------------------------------------------------------- |0 | SELECT STATEMENT|| 49075 |3786K|52(2)| 00:00:01 | |*1 |TABLE ACCESS FULL| TEST | 49075 |3786K|52(2)| 00:00:01 | -------------------------------------------------------------------------- Predicate Information (identified by operation id): --------------------------------------------------- 1 - filter("ID"=1)Note ----- - dynamic sampling used for this statementStatistics ---------------------------------------------------------- 32recursive calls 0db block gets 3644consistent gets 0physical reads 0redo size 1689175bytes sent via SQL*Net to client 37363bytes received via SQL*Net from client 3356SQL*Net roundtrips to/from client 0sorts (memory) 0sorts (disk) 50314rows processedSQL> select * from test where id =100; Execution Plan ---------------------------------------------------------- Plan hash value: 2473784974---------------------------------------------------------------------------------------- | Id| Operation| Name| Rows| Bytes | Cost (%CPU)| Time| ---------------------------------------------------------------------------------------- |0 | SELECT STATEMENT||1 |79 |2(0)| 00:00:01 | |1 |TABLE ACCESS BY INDEX ROWID| TEST|1 |79 |2(0)| 00:00:01 | |*2 |INDEX RANGE SCAN| IDX_TEST |1 ||1(0)| 00:00:01 | ---------------------------------------------------------------------------------------- Predicate Information (identified by operation id): --------------------------------------------------- 2 - access("ID"=100)Note ----- - dynamic sampling used for this statementStatistics ---------------------------------------------------------- 9recursive calls 0db block gets 73consistent gets 0physical reads 0redo size 588bytes sent via SQL*Net to client 469bytes received via SQL*Net from client 2SQL*Net roundtrips to/from client 0sorts (memory) 0sorts (disk) 1rows processed

仅此一项就可以看出为什么ORACLE极力推荐使用CBO,从ORACLE 10g开始不支持RBO的缘故。所谓长江后浪推前浪,前浪死在沙滩上。
CBO知识点的总结 CBO优化器根据SQL语句生成一组可能被使用的执行计划,估算出每个执行计划的代价,并调用计划生成器(Plan Generator)生成执行计划,比较执行计划的代价,最终选择选择一个代价最小的执行计划。查询优化器由查询转换器(Query Transform)、代价估算器(Estimator)和计划生成器(Plan Generator)组成。
CBO优化器组件 CBO由以下组件构成:
  • 查询转化器(Query Transformer)
    查询转换器的作用就是等价改变查询语句的形式,以便产生更好的执行计划。它决定是否重写用户的查询(包括视图合并、谓词推进、非嵌套子查询/子查询反嵌套、物化视图重写),以生成更好的查询计划。
The input to the query transformer is a parsed query, which is represented by a set of query blocks. The query blocks are nested or interrelated to each other. The form of the query determines how the query blocks are interrelated to each other. The main objective of the query transformer is to determine if it is advantageous to change the form of the query so that it enables generation of a better query plan. Several different query transformation techniques are employed by the query transformer, including: ■ View Merging ■ Predicate Pushing ■ Subquery Unnesting ■ Query Rewrite with Materialized Views Any combination of these transformations can be applied to a given query.

  • 代价评估器(Estimator)
    评估器通过复杂的算法结合来统计信息的三个值来评估各个执行计划的总体成本:选择性(Selectivity)、基数(Cardinality)、成本(Cost)
  • 计划生成器(Plan Generator)
    计划生成器就是生成大量的执行计划,然后选择其总体代价或总体成本最低的一个执行计划。
    计划生成器会考虑可能的访问路径(Access Path)、关联方法和关联顺序,生成不同的执行计划,让查询优化器从这些计划中选择出执行代价最小的一个计划。
    由于不同的访问路径、连接方式和连接顺序可以组合,虽然以不同的方式访问和处理数据,但是可以产生同样的结果
ORACLE优化器之RBO与CBO
文章图片

查看ORACLE优化器
SQL> show parameter optimizer_mode; NAMETYPEVALUE --------------------------- ----------- ----------------- optimizer_modestringALL_ROWS

修改ORACLE优化器 ORACLE 10g 优化器可以从系统级别、会话级别、语句级别三种方式修改优化器模式,非常方便灵活。
其中optimizer_mode可以选择的值有: first_rows_n,all_rows. 其中first_rows_n又有first_rows_1000, first_rows_100, first_rows_10, first_rows_1
在Oracle 9i中,优化器模式可以选择first_rows_n,all_rows, choose, rule 等模式:
Rule: 基于规则的方式。
Choose:指的是当一个表或或索引有统计信息,则走CBO的方式,如果表或索引没统计信息,表又不是特别的小,而且相应的列有索引时,那么就走索引,走RBO的方式。
First Rows:它与Choose方式是类似的,所不同的是当一个表有统计信息时,它将是以最快的方式返回查询的最先的几行,从总体上减少了响应时间。
All Rows: 10g中的默认值,也就是我们所说的Cost的方式,当一个表有统计信息时,它将以最快的方式返回表的所有的行,从总体上提高查询的吞吐
虽然Oracle 10g中不再支持RBO,Oracle 10g官方文档关于optimizer_mode参数的只有first_rows和all_rows.但是依然可以设置 optimizer_mode为rule或choose,估计是ORACLE为了过渡或向下兼容考虑。如下所示。
系统级别
SQL> alter system set optimizer_mode=rule scope=both; System altered.SQL> show parameter optimizer_modeNAMETYPEVALUE -------------------------------- ----------- ----------------------- optimizer_modestringRULE

【ORACLE优化器之RBO与CBO】会话级别
会话级别修改优化器模式,只对当前会话有效,其它会话依然使用系统优化器模式。
SQL> alter session set optimizer_mode=first_rows_100; Session altered.

语句级别
语句级别通过使用提示hints来实现。
SQL> select /*+ rule */ * from dba_objects where rownum <= 10;

补充说明:
在Oracle中,是通过为语句添加 Hints(提示)来实现干预优化器优化的目的,Oracle Hints是一种机制,用来告诉优化器按照我们的告诉它的方式生成执行计划。
Hints语法详细解读请参考:http://blog.csdn.net/wb96a1007/article/details/7098687

    推荐阅读