[oracle实验]混合使用不同提示的hint

概述

最近比较忙,介绍一个之前做的简单点的实验,混合使用不同提示的hint,大家有空可以跟着做下。


基础环境准备

drop table t1;
create table t1 as select * from dba_objects;
insert into t1 select * from t1;
insert into t1 select * from t1;
commit;
select count(*) from t1; =>349300
create index idx_t1 on t1(object_id);
update t1 set object_id=1 where rownum<320000;
commit;

[oracle实验]混合使用不同提示的hint

/手动修改了OBJECT_ID的值,将表中绝大多数记录的OBJECT_ID设置为1

select count(*) from t1 where object_id=1; =>319999

//收集表的统计信息,注意此时也收集了相关对象--索引的统计信息

exec dbms_stats.gather_table_stats(ownname=>'SCOTT',tabname=>'T1',estimate_percent=>100,method_opt=>'for columns size auto object_id',cascade=>true);

[oracle实验]混合使用不同提示的hint

/*

查看当前索引的聚簇因子为5531,聚簇因子反映了索引字段的顺序和表中数据存储的有序关系。聚簇因子越小,说明索引字段顺序与表中数据存储顺序一致性越高;反之,则一致性越低,即越无序。

*/

select clustering_factor from dba_indexes where index_name='IDX_T1'; =》5531

//这里手动修改了聚簇因子为10000,手动修改统计信息,是一种常见优化手段,便于分析问题

exec dbms_stats.set_index_stats(ownname=>'SCOTT',indname=>'IDX_T1',clstfct=>10000,no_invalidate=>false);
select clustering_factor from dba_indexes where index_name='IDX_T1';

[oracle实验]混合使用不同提示的hint


测试sql(默认情况)

默认情况采用索引扫描,因为上面修改了索引的聚簇因子,大大增加了索引扫描的成本,所以这里选择全表扫描

set autotrace traceonly;
select object_name,object_id from t1 where object_id=1;

[oracle实验]混合使用不同提示的hint

测试sql(first_rows(10))

first_rows(10)作用是优先返回10条记录,在使用hint后,oracle认为此时扫描索引IDX_T1能够以最短响应时间返回where条件“object_id=1”的头10条记录,所以这里用了索引范围扫描。

select /*+ first_rows(10) */ object_name,object_id from t1 where object_id=1;

[oracle实验]混合使用不同提示的hint


测试sql(first_rows(9))

first_rows(9)作用是优先返回9条记录,与上面变化是优化器对基数的估算不同,注意执行计划中的rows部分,从first_rows(10)的11变成了9。

select /*+ first_rows(9) */ object_name,object_id from t1 where object_id=1;

[oracle实验]混合使用不同提示的hint

[oracle实验]混合使用不同提示的hint


测试SQL(rule)

注意执行计划中的关键字“rule based...”,显示的具体执行步骤中并没有“cost”列,这说明RULE起作用了,现在是RBO。

select /*+ rule */ object_name,object_id from t1 where object_id=1;

[oracle实验]混合使用不同提示的hint


测试SQL(rule+parallel)

输出中包含了“cost”列,表示SQL在解析时使用CBO,说明如果目标SQL使用了并行执行,其中的RULE HINT将会失效,此时Oracle自动启用CBO.

alter table t1 parallel;
select /*+ rule */ object_name,object_id from t1 where object_id=1;

[oracle实验]混合使用不同提示的hint


后面小编会分享更多DBA方面的干货,感兴趣的朋友走一波关注哩~

[oracle实验]混合使用不同提示的hint

相关推荐