oracle 实现基于函数的索引
admin
2023-05-13 05:01:42
0

使用场景:
当一个查询运行很慢。通过检查where子句,发现其中的一列应用了sql lower函数,lower函数
阻止使用该列上现有的索引。你想要创建一个基于函数索引来支持这个查询,如下

SQL> select index_name,column_name from user_ind_columns where table_name='T1';

INDEX_NAME                COLUMN_NAME
------------------------- ------------------------------
T1_PK                     OBJECT_ID

SQL> set autotrace trace explain;
SQL> select * from t1 where lower(object_name)='i_undo1';

执行计划
----------------------------------------------------------
Plan hash value: 3617692013

--------------------------------------------------------------------------
| Id  | Operation         | Name | Rows  | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------
|   0 | SELECT STATEMENT  |      |   908 |   101K|   436   (1)| 00:00:01 |
|*  1 |  TABLE ACCESS FULL| T1   |   908 |   101K|   436   (1)| 00:00:01 |
--------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter(LOWER("OBJECT_NAME")='i_undo1')
从以上可以看出即使该表中有索引也没有使用。

解决方案
1、创建一个基于函数的索引
2、如果使用oracle database 11g或更高版本,创建一个索引虚拟列

下面实现基于函数的索引

创建索引可以通过以下方式来估计索引所使用空间以及所需要分配的空间

SQL> set serveroutput on
SQL> var used_bytes number;
SQL> var allo_bytes number;
SQL> exec dbms_space.create_index_cost('create index t1_object_name on t1(lower(object_name))',:used_bytes,:allo_bytes);
 
PL/SQL procedure successfully completed
used_bytes
---------
2269350
allo_bytes
---------
4194304

SQL>create index idx_lower on t1(lower(object_name)) tablespace index_nocompress;

SQL> select * from t1 where lower(object_name)='i_undo1';

执行计划
----------------------------------------------------------
Plan hash value: 2274688371

-------------------------------------------------------------------------------------------------
| Id  | Operation                           | Name      | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT                    |           |   908 |   101K|   193   (0)| 00:00:01 |
|   1 |  TABLE ACCESS BY INDEX ROWID BATCHED| T1        |   908 |   101K|   193   (0)| 00:00:01 |
|*  2 |   INDEX RANGE SCAN                  | IDX_LOWER |   363 |       |     3   (0)| 00:00:01 |
-------------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access(LOWER("OBJECT_NAME")='i_undo1')

注意:不能直接修改一个创建了基于函数索引的列。需要先删除索引,然后修改列,最后再重建索引。不然会报ora-30556错误(在要修改的列上已定义函数索引或位图联接索引)

查看基于函数的索引定义dba/all/user_ind_expressions

SQL> select index_name,column_expression from user_ind_expressions;

INDEX_NAME                COLUMN_EXPRESSION
------------------------- --------------------------------------------------
IDX_LOWER                 LOWER("OBJECT_NAME")

接着实现在虚拟列创建一个索引

使用场景

现在正使用一个基于函数的索引,但想要获得更好的性能,想将基于函数的索引替换为一个虚拟列,然后在虚拟列上创建索引(需要11g环境或更高版本)。

SQL>alter table t1 add(lower_object_name generated always as (lower(object_name)) virtual);
SQL>create index idx_lower on t1(lower_object_name) tablespace index_nocompress;


 

 

相关内容

热门资讯

时隔两年,农业农村部再调整这一... ‍‍5月14日,农业农村部发布消息称,近日,农业农村部印发《生猪产能综合调控实施方案(2026年修订...
官宣!天风证券迎来新任总裁 【大河财立方消息】5月14日晚间,天风证券公告,近期收到王琳晶递交的书面辞职报告。因个人原因,王琳晶...
特朗普访华,欢迎晚宴上有哪些中... 【大河财立方 记者 闫文瑞】 5月14日晚,欢迎美国总统特朗普访华宴会上,多位中国领军企业家受邀出席...
央行缩量续作3000亿元买断式... 【大河财立方 记者 杨萨】 5月14日,央行公告称,为保持银行体系流动性充裕,2026年5月15日,...
京东物流切入商业航天场景 【大河财立方消息】 据京东物流黑板报,5月14日,蓝箭航天朱雀二号改进型遥五运载火箭(ZQ-2E Y...
首超3000亿元!河南前四个月... 【大河财立方 记者 丁洋涛】今年1—4月,河南外贸规模历史同期首次超3000亿元。5月14日,记者从...
菲律宾将调查“参议院枪击事件” 【环球时报驻菲律宾特派记者 樊帆 郑真】菲律宾参议院大楼13日晚传出数十声枪响和人群惊慌失措的尖叫声...
五个快递员的故事里,藏着你我生... 【大河财立方 记者 丁洋涛 裴熔熔 北京报道】 5月14日下午,国务院新闻办发布厅,一场特殊的见面会...
M2同比增长8.6%!央行发布... 【大河财立方消息】5月14日,中国人民银行发布2026年4月金融统计数据报告。一、社会融资规模存量同...
河南完成发行259.35亿元地... 【大河财立方消息】河南省财政厅消息,5月14日,河南省在深圳成功发行政府债券259.35亿元。其中:...