总结:
(1)alter index ii storage(next 40)10g的不可以改变索引的存储参数,但是11g开始可以改变索引的部分存储参数,其中INITIAL
和 MINEXTENTS
不可以被修改
(2) alter index ii rebuild online tablespace test 可以改变存储参数,改变索引的表空间,减少表空间的碎片,在索引比较大的时候可以用paralle+nologging 在加快索引的建立
(3) alter index index_name unusable;索引会被优化器忽略,并且dml操作也不会维护这个索引,索引必须重建或者删除.10g环境下索引状态变成unusable时,段 的空间没有释放,11g环境索引状态变成unusable时,段空间释放. 唯一索引变成unusable状态后不能插入属于
分区索引修改成unusable
ALTER INDEX i_emp_ename MODIFY PARTITION p2_i_emp_ename UNUSABLE;
(4)索引的disable状态只是针对函数索引来设置的,其他索引都不能设置成disable,如果函数索引变成disable状态后,不能插入数据。
(5)invisible index 会被优化器所忽略,但是可以设置参数OPTIMIZER_USE_INVISIBLE_INDEXES,优化器会索引,索引变成invisible,不影响数据的插入
(6)monriting index usage只能监控此索引是否被使用过,每次 MONITORING USAGE开启后,
V$OBJECT_USAGE
视图就会初始化一次重设, 之前的清理就被重 设.在 MONITORING 时,如果索引被使用,
里就会显示此索引被使用V$OBJECT_USAGE
遗留问题:
(1)在11g中为什么要修改存储的参数 alter index ii storage(next 40),什么情况下要修改这个参数呢?
(2)为什么要把函数索引变成disable状态,什么情况下disable 函数索引
(3)11g后索引变成unusable 后,段空间被释放,那么rebuild index 怎么利用现存的索引重建的呢
1、如果你想alter index ,那么这个索引要么建立在你的用户下,要么你必须有 ALTER ANY INDEX
的系统权限,当你有alter any index权限后,你可以对索引进行如下更改,但是不能改变索引列的结构:
Rebuild or coalesce an existing index
Deallocate unused space or allocate a new extent
Specify parallel execution (or not) and alter the degree of parallelism
Alter storage parameters or physical attributes
Specify LOGGING
or NOLOGGING
Enable or disable key compression
Mark the index unusable
Make the index invisible
Rename the index
Start or stop the monitoring of index usage
1.2、改变索引的存储特征
通过alter index 去改变包含主键、唯一约束,普通索引,但是 不能通过alter index 修改索引的存储参数
INITIAL
和MINEXTENTS
。版本10.2.0.1.0:
SQL>create table t1 as select * from emp;
表已创建。
SQL> create index ii on t1(empno);
SQL>select dbms_metadata.get_ddl('INDEX','II','SCOTT') from dual;
DBMS_METADATA.GET_DDL('INDEX','II','SCOTT')
--------------------------------------------------------------------------------
CREATE INDEX "SCOTT"."II" ON "SCOTT"."T1" ("EMPNO")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0
但是
SQL> ALTER INDEX scott.ii STORAGE (NEXT 40);
ALTER INDEX scott.ii STORAGE (NEXT 40)
*
ERROR at line 1:
ORA-25150: ALTERING of extent parameters not permitted
SQL> alter index ii storage(next 8192);
alter index ii storage(next 8192)
*
ERROR at line 1:
ORA-25150: ALTERING of extent parameters not permitted
SQL> alter index ii storage(next 16384);
alter index ii storage(next 16384)
*
ERROR at line 1:
ORA-25150: ALTERING of extent parameters not permitted
当版本是11.2.0.1.0 :
SQL> create table t1 as select * from emp;
表已创建。
SQL> create index ii on t1(empno);
索引已创建。
SQL> select dbms_metadata.get_ddl('INDEX','II','SCOTT') from dual;
DBMS_METADATA.GET_DDL('INDEX','II','SCOTT')
----------------------------------------------------------------------------
CREATE INDEX "SCOTT"."II" ON "SCOTT"."T1" ("EMPNO")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 65536 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0
SQL> alter index ii storage(next 40);
索引已更改。
SQL> select dbms_metadata.get_ddl('INDEX','II','SCOTT') from dual;
DBMS_METADATA.GET_DDL('INDEX','II','SCOTT')
------------------------------------------------------------------------------
CREATE INDEX "SCOTT"."II" ON "SCOTT"."T1" ("EMPNO")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 65536 NEXT 8192 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 FR
SQL> alter index ii storage(next 16384);
索引已更改。
SQL> select dbms_metadata.get_ddl('INDEX','II','SCOTT') from dual;
DBMS_METADATA.GET_DDL('INDEX','II','SCOTT')
------------------------------------------------------------------------------
CREATE INDEX "SCOTT"."II" ON "SCOTT"."T1" ("EMPNO")
PCTFREE 10 INITRANS 2 MAXTRANS 255 COMPUTE STATISTICS
STORAGE(INITIAL 65536 NEXT 16384 MINEXTENTS 1 MAXEXTENTS 2147483645
PCTINCREASE 0 F
结论:通过实验我们看到,在10.2.0.1.0 版本的时候是不允许修改索引的存储参数的,但是到了11.2.0.1.0,此参数是可以修改的
1.3、Rebuilding an Existing Index
一个现有的索引可以重建,也可以coalesce,也可以删除新建,考虑其中的代价,来决定是那种方式。
当你拿现有索引当成源,重新建索引的时候,可以改变存储的参数,改变表空间,减少现有索引中的碎片,对比删除索引重建,rebuild会性能好一些。rebuild其他用户的索引需要的权限:CREATE ANY TABLE、 CREATE ANY INDEX
SQL> alter index ii rebuild online tablespace test;
Index altered.Index altered.最好是加上online 参数,否则其他session就不能对表进行dml操作。对于比较大的索引可以加上 paralle +nologging 加快索引的重建。
alter index ii rebuild online nologging tablespace test parallel 2;alter index ii logging;alter index ii noparallel;online rebuild index 有个限制就是不能超过最大键值的长度ORA-1450 (maximum key length exceeded) ,如果报错可以采用offline, coalescing, or dropping and re-creating。
当你把一个索引的状态由valid,变成unusable时,这个索引会被优化器忽略,并且dml操作也不会维护这个索引。当你把一个分区的分区索引变成unusable时,其他分区仍然是valid状态。在使用这个索引前你必须删除重建或者rebuild
当版本是10G:查看索引的状态:SQL> SELECT INDEX_NAME AS "INDEX OR PART NAME", STATUS FROM USER_INDEXESUNION ALLSELECT PARTITION_NAME AS "INDEX OR PART NAME", STATUS FROM USER_IND_PARTITIONS;INDEX OR PART NAME STATUS------------------------------ --------II VALIDSQL> alter index ii unusable;Index altered.SQL> SELECT INDEX_NAME AS "INDEX OR PART NAME", STATUS FROM USER_INDEXES
UNION ALL
SELECT PARTITION_NAME AS "INDEX OR PART NAME", STATUS FROM USER_IND_PARTITIONS;UNION ALLSELECT PARTITION_NAME AS "INDEX OR PART NAME", STATUS FROM USER_IND_PARTITIONS;INDEX OR PART NAME STATUS------------------------------ --------II UNUSABLESQL> select segment_name,bytes from user_segments where segment_name='II';SEGMENT_NAME BYTES--------------------------------------------------------------------------------- ----------II 4194304当11g环境:SQL> select status from dba_indexes where table_name='T1';STATUS--------VALIDSQL> alter index ii unusable;索引已更改。SQL> select status from dba_indexes where table_name='T1';STATUS--------UNUSABLESQL> select segment_name,segment_type,extent_id,blocks from dba_extents where segment_name='II';未选定行
如果是分区索引这样修改:ALTER INDEX i_emp_ename MODIFY PARTITION p2_i_emp_ename UNUSABLE;
SQL> create unique index ii on t1(empno);
索引已创建。
INDEX_NAME INDEX_TYPE VISIBILIT STATUS UNIQUENES------------------------------ --------------------------- --------- -------- ---------II NORMAL VISIBLE VALID UNIQUEIND FUNCTION-BASED NORMAL VISIBLE VALID NONUNIQUESQL> alter index ii unusable;索引已更改。SQL> select a.index_name,a.index_type,a.visibility,status,a.uniqueness from dba_indexes a where table_name='T1';INDEX_NAME INDEX_TYPE VISIBILIT STATUS UNIQUENES------------------------------ --------------------------- --------- -------- ---------II NORMAL VISIBLE UNUSABLE UNIQUEIND FUNCTION-BASED NORMAL VISIBLE VALID NONUNIQUESQL> insert into t1 select *From emp;insert into t1 select *From emp*第 1 行出现错误:ORA-01502: 索引 'SCOTT.II' 或这类索引的分区处于不可用状态
DISABLE
applies only to a function-based index. This clause lets you disable the use of a function-based index. You might want to do so, for example, while working on the body of the function. Afterward you can either rebuild the index or specify anotherALTER
INDEX
statement with theENABLE
keywordSQL> create unique index uq on t1(upper(ename));索引已创建。SQL> select INDEX_NAME,status,funcidx_status from dba_indexes where table_name='T1';INDEX_NAME STATUS FUNCIDX_------------------------------ -------- --------II VALIDUQ VALID ENABLEDSQL> select INDEX_NAME,status,funcidx_status from dba_indexes where table_name='T1';INDEX_NAME STATUS FUNCIDX_------------------------------ -------- --------II VALIDUQ VALID DISABLEDSQL> insert into t1 select *From t1;insert into t1 select *From t1*第 1 行出现错误:ORA-30554: 基于函数的索引SCOTT.UQ被禁用SQL> drop index uq;索引已删除。SQL> create index ind on t1(lower(ename));索引已创建。
SQL> select INDEX_NAME,status,funcidx_status from dba_indexes where table_name='T1';INDEX_NAME STATUS FUNCIDX_------------------------------ -------- --------II VALIDIND VALID ENABLEDSQL> alter index ind disable;索引已更改。SQL> select INDEX_NAME,status,funcidx_status from dba_indexes where table_name='T1';INDEX_NAME STATUS FUNCIDX_------------------------------ -------- --------II VALIDIND VALID DISABLEDSQL> insert into t1 select *From emp;insert into t1 select *From emp*第 1 行出现错误:ORA-30554: 基于函数的索引SCOTT.IND被禁用SQL> alter index ind enable;索引已更改。SQL> alter index ind unusable;索引已更改。SQL> insert into t1 select *From emp;已创建12行。
不可见的索引是会被优化器所忽略,除非你在session级别或者system级别设置了参数OPTIMIZER_USE_INVISIBLE_INDEXES
=true,让索引不可见是为了让索引unusable或者要删除索引而考虑。分区索引是不能不可见的,否则会报错。SQL> select a.index_name,a.index_type,a.visibility,status from dba_indexes a where table_name='T1';
INDEX_NAME INDEX_TYPE VISIBILIT STATUS
------------------------------ --------------------------- --------- --------
II NORMAL VISIBLE VALID
IND FUNCTION-BASED NORMAL VISIBLE VALID
SQL> select *from t1 where empno=7788;
执行计划
----------------------------------------------------------
Plan hash value: 2627886109
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 87 | 1 (0)| 00:00:01 |
| 1 | TABLE ACCESS BY INDEX ROWID| T1 | 1 | 87 | 1 (0)| 00:00:01 |
|* 2 | INDEX RANGE SCAN | II | 1 | | 1 (0)| 00:00:01 |
------------------------------------------------------------------------------------
SQL> alter index ii invisible;
索引已更改。
SQL> alter index ind invisible;
索引已更改。
SQL> select a.index_name,a.index_type,a.visibility,status from dba_indexes a where table_name='T1';
INDEX_NAME INDEX_TYPE VISIBILIT STATUS
------------------------------ --------------------------- --------- --------
II NORMAL INVISIBLE VALID
IND FUNCTION-BASED NORMAL INVISIBLE VALID
---可以正常插入数据
SQL> insert into t1 select *From emp;
已创建12行。
SQL> select *from t1 where empno=7788;
执行计划
----------------------------------------------------------
Plan hash value: 3617692013
--------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
--------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 87 | 3 (0)| 00:00:01 |
|* 1 | TABLE ACCESS FULL| T1 | 1 | 87 | 3 (0)| 00:00:01 |
--------------------------------------------------------------------------
ii invisible,优化器忽略了索引 ii
如果修改参数:
SQL> alter system set optimizer_use_invisible_indexes=true;
系统已更改。
SQL> select *from t1 where empno=7788;
执行计划
----------------------------------------------------------
Plan hash value: 2627886109
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 87 | 1 (0)| 00:00:01 |
| 1 | TABLE ACCESS BY INDEX ROWID| T1 | 1 | 87 | 1 (0)| 00:00:01 |
|* 2 | INDEX RANGE SCAN | II | 1 | | 1 (0)| 00:00:01 |
------------------------------------------------------------------------------------
SQL> alter index ii visible;
索引已更改。
SQL> alter index ind visible;
索引已更改。
SQL> select a.index_name,a.index_type,a.visibility,status from dba_indexes a where table_name='T1';
INDEX_NAME INDEX_TYPE VISIBILIT STATUS
------------------------------ --------------------------- --------- --------
II NORMAL VISIBLE VALID
IND FUNCTION-BASED NORMAL VISIBLE VALID
结论 :invisible index 会被优化器所忽略,但是可以设置参数OPTIMIZER_USE_INVISIBLE_INDEXES,优化器会索引,索引变成invisible,不影响数据的插入
SQL> alter index ii rename to ind_i;
索引已更改。
SQL> select a.index_name,a.index_type,a.visibility,status from dba_indexes a where table_name='T1';
INDEX_NAME INDEX_TYPE VISIBILIT STATUS
------------------------------ --------------------------- --------- --------
IND_I NORMAL VISIBLE VALID
IND FUNCTION-BASED NORMAL VISIBLE VALID
1.8、Monitoring Index Usage
此方法只能监控此索引是否被使用过,每次
MONITORING USAGE开启后,
V$OBJECT_USAGE
视图就会初始化一次重设, 之前的清理就被重设.在MONITORING 时,如果索引被使用,
里就会显示此索引被使用.V$OBJECT_USAGE
SQL> drop index ind_i;
索引已删除。
SQL> create index ii on t1(empno);
索引已创建。
SQL> alter index ii monitoring usage;
索引已更改。
SQL> select *From v$object_usage;
INDEX_NAME TABLE_NAME MON USE START_MONITORING END_MONITORING
------------------------------ ------------------------------ - ------------- --- ------------------- -------------------
II T1 YES NO 05/21/2013 15:55:19
SQL> select *From t1 where empno=7788;
执行计划
----------------------------------------------------------
Plan hash value: 2627886109
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 87 | 1 (0)| 00:00:01 |
| 1 | TABLE ACCESS BY INDEX ROWID| T1 | 1 | 87 | 1 (0)| 00:00:01 |
|* 2 | INDEX RANGE SCAN | II | 1 | | 1 (0)| 00:00:01 |
------------------------------------------------------------------------------------
SQL> select *From v$object_usage;
INDEX_NAME TABLE_NAME MON USE START_MONITORING END_MONITORING
--------------------------- ------------------------------ --- --- ------------------- -------------------
II T1 YES YES 05/21/2013 15:55:19
参考:http://docs.oracle.com/cd/E11882_01/server.112/e25494/indexes004.htm#ADMIN13283