Oracle虚拟索引

前端之家收集整理的这篇文章主要介绍了Oracle虚拟索引前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

从9.2版本开始Oracle引入了虚拟索引的概念,虚拟索引是一个“伪造”的索引,它的定义只存在数据字典中并有存在相关的索引段。虚拟索引是为了在不真正创建索引的情况下,验证如果使用索引sql执行计划是否改变,执行效率是否能得到提高。

本文在11.2.0.4版本中测试使用虚拟索引

1、创建测试表

ZX@orcl>createtabletest_tasselect*fromdba_objects;

Tablecreated.

ZX@orcl>selectcount(*)fromtest_t;

COUNT(*)
----------
86369

2、查看一个sql的执行计划,由于没有创建索引,使用TABLE ACCESS FULL访问表

ZX@orcl>setautotracetraceonlyexplain
ZX@orcl>selectobject_namefromtest_twhereobject_id=123;

ExecutionPlan
----------------------------------------------------------
Planhashvalue:2946757696

----------------------------------------------------------------------------
|Id|Operation	|Name|Rows|Bytes|Cost(%cpu)|Time	|
----------------------------------------------------------------------------
|0|SELECTSTATEMENT|	|	14|1106|344(1)|00:00:05|
|*1|TABLEACCESSFULL|TEST_T|	14|1106|344(1)|00:00:05|
----------------------------------------------------------------------------

PredicateInformation(identifiedbyoperationid):
---------------------------------------------------

1-filter("OBJECT_ID"=123)

Note
-----
-dynamicsamplingusedforthisstatement(level=2)

3、创建虚拟索引,数据字典中有这个索引的定义但是并没有实际创建这个索引段

ZX@orcl>setautotraceoff
ZX@orcl>createindexidx_virtualontest_t(object_id)nosegment;

Indexcreated.

ZX@orcl>selectobject_name,object_typefromuser_objectswhereobject_name='IDX_VIRTUAL';

OBJECT_NAME															OBJECT_TYPE
---------------------------------------------------------------------------------------------------------------------------------------------------
IDX_VIRTUAL															INDEX

ZX@orcl>selectsegment_name,tablespace_namefromuser_segmentswheresegment_name='IDX_VIRTUAL';

norowsselected

4、再次查看执行计划

ZX@orcl>setautotracetraceonlyexplain
ZX@orcl>selectobject_namefromtest_twhereobject_id=123;

ExecutionPlan
----------------------------------------------------------
Planhashvalue:2946757696

----------------------------------------------------------------------------
|Id|Operation	|Name|Rows|Bytes|Cost(%cpu)|Time	|
----------------------------------------------------------------------------
|0|SELECTSTATEMENT|	|	14|1106|344(1)|00:00:05|
|*1|TABLEACCESSFULL|TEST_T|	14|1106|344(1)|00:00:05|
----------------------------------------------------------------------------

5、我们看到执行计划并没有使用上面创建的索引,要使用虚拟索引需要设置参数

ZX@orcl>altersessionset"_use_nosegment_indexes"=true;

Sessionaltered.

6、再次查看执行计划,可以看到执行计划选择了虚拟索引,而且时间也缩短了。

ZX@orcl>selectobject_namefromtest_twhereobject_id=123;

ExecutionPlan
----------------------------------------------------------
Planhashvalue:1533029720

-------------------------------------------------------------------------------------------
|Id|Operation		|Name	|Rows|Bytes|Cost(%cpu)|Time	|
-------------------------------------------------------------------------------------------
|0|SELECTSTATEMENT	|		|14|1106|	5(0)|00:00:01|
|1|TABLEACCESSBYINDEXROWID|TEST_T	|14|1106|	5(0)|00:00:01|
|*2|INDEXRANGESCAN	|IDX_VIRTUAL|315|	|	1(0)|00:00:01|
-------------------------------------------------------------------------------------------

PredicateInformation(identifiedbyoperationid):
---------------------------------------------------

2-access("OBJECT_ID"=123)

Note
-----
-dynamicsamplingusedforthisstatement(level=2)

从上面的执行计划可以看出创建这个索引会起到优化的效果,这个功能在大表建联合索引优化能起到很好的做作用,可以测试多个列组合哪个组合效果最好,而不需要实际每个组合都创建一个大索引。

7、删除虚拟索引

ZX@orcl>dropindexidx_virtual;

Indexdropped.


MOS文档:Virtual Indexes (文档 ID 1401046.1)

猜你在找的Oracle相关文章