---------------------------------------2014-05-18-------------------------------------------------------------------------------------------------------------------
时间总归是要打败你我
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
今天必须弄清楚本地索引、全局索引、前缀索引、非前缀索引这些名词间的关系
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
分区索引分为本地(local index)索引和全局索引(global index)。
其中本地索引又可以分为有前缀(prefix)的索引和无前缀(nonprefix)的索引。而全局索引目前只支持有前缀的索引。
B树索引和位图索引都可以分区,但是HASH索引不可以被分区。
位图索引必须是本地索引。下面就介绍本地索引以及全局索引各自的特点来说明区别;
一、本地索引特点:
1. 本地索引一定是分区索引,分区键等同于表的分区键,分区数等同于表的分区数,一句话,本地索引的分区机制和表的分区机制一样。
2. 如果本地索引的索引列以分区键开头,则称为前缀局部索引。
3. 如果本地索引的列不是以分区键开头,或者不包含分区键列,则称为非前缀索引。
4. 前缀和非前缀索引都可以支持索引分区消除,前提是查询的条件中包含索引分区键。
5. 本地索引只支持分区内的唯一性,无法支持表上的唯一性,因此如果要用本地索引去给表做唯一性约束,则约束中必须要包括分区键列。
6. 本地分区索引是对单个分区的,每个分区索引只指向一个表分区,全局索引则不然,一个分区索引能指向n个表分区,
同时,一个表分区,也可能指向n个索引分区,对分区表中的某个分区做truncate或者move,shrink等,可能会影响到n个全局索引分区,正因为这点,本地分区索引具有更高的可用性。
7. 位图索引只能为本地分区索引。
8. 本地索引多应用于数据仓库环境中。
本地索引:创建了一个分区表后,如果需要在表上面创建索引,并且索引的分区机制和表的分区机制一样,那么这样的索引就叫做本地分区索引。(这里的分区机制指的是什么??)
本地索引是由ORACLE自动管理的,它分为有前缀的本地索引和无前缀的本地索引。什么叫有前缀的本地索引?
有前缀的本地索引就是包含了分区键,并且将其作为引导列的索引。什么叫无前缀的本地索引?无前缀的本地索引就是没有将分区键的前导列作为索引的前导列的索引。下面举例说明:
创建分区表:以id作为分区条件
create table test (id number,data varchar2(100))
partition by RANGE (id)
(
partition p1 values less than (1000) tablespace users,
partition p2 values less than (2000) tablespace users,
partition p3 values less than (maxvalue) tablespace users
);
创建本地索引:以分区条件id作为索引分区
create index i_id on test(id) local; 因为id是分区键,所以这样就创建了一个有前缀的本地索引。
select dbms_metadata.get_ddl('INDEX','I_ID','KING') index_name FROM DUAL;
这条语句查询出来的结果和我执行的语句是一样的,并没有看到转换后的分区语句。
创建无前缀的本地分区索引:
create index i_data on test(data) local;因为data不是分区键,所以这样就创建了一个无前缀的本地索引。(也就是说,一张分区表上面建的分区索引,只要不是以表分区条件作为引导列的索引
都是非前缀本地分区索引)
select dbms_metadata.get_ddl('INDEX','I_DATA','KING') index_name FROM DUAL;
INDEX_NAME
------------------------------------------------------------------
CREATE INDEX "KING"."I_DATA" ON "KING"."TEST" ("DATA")
PCTFREE 10 INITRANS
理想状态是应该看到如下语句:
CREATE INDEX "KING"."I_DATA" ON "KING"."TEST" ("DATA") LOCAL
(PARTITION "P1" TABLESPACE "USERS" ,PARTITION "P2" TABLESPACE "USERS" , PARTITION "P3" TABLESPACE "USERS" );
这种情况下方便我们对本地索引进行理解,比如上面我对本地分区索引的分区机制和表分区机制相同的不理解,从上面的语句就可以得到答复。
索引的分区就按照表的分区进行的,
表的分区条件是ID,相对应的ID值都落在了P1,P2,P3这三个分区里面。
当我们再创建DATA这个非前缀本地索引后,其实DATA的范围域就是根据ID的范围域来划分的。比如P1分区对应着ID小于1000的值,那么DATA在P1分区ID值就是对应着小于1000的值。
从user_part_indexes视图也可以证明刚才创建的索引,一个是有前缀的,一个是无前缀的
select index_name,table_name,partitioning_type,locality,ALIGNMENT from user_part_indexes;
index_name table_name partitioning_type locality ALIGNMENT
2 I_ID TEST RANGE LOCAL PREFIXED
3 I_DATA TEST RANGE LOCAL NON_PREFIXED
二、全局索引特点:
1. 全局索引的分区键和分区数和表的分区键和分区数可能都不相同,表和全局索引的分区机制不一样。()
2. 全局索引可以分区,也可以是不分区索引,全局索引必须是前缀索引,即全局索引的索引列必须是以索引分区键作为其前几列。
3. 全局分区索引的索引条目可能指向若干个分区,因此,对于全局分区索引,即使只截断一个分区中的数据,都需要rebulid若干个分区甚至是整个索引。
4. 全局索引多应用于oltp系统中。
5. 全局分区索引只按范围或者散列hash分区,hash分区是10g以后才支持。
6. oracle9i以后对分区表做move或者truncate的时可以用update global indexes语句来同步更新全局分区索引,用消耗一定资源来换取高度的可用性。
7. 表用a列作分区,索引用b做局部分区索引,若where条件中用b来查询,那么oracle会扫描所有的表和索引的分区,成本会比分区更高,此时可以考虑用b做全局分区索引。
另外oracle不会自动的维护全局分区索引,当我们在对表的分区做修改之后,如果执行修改的语句不加上update global indexes的话,那么索引将不可用。
以上面创建的分区表test为例,讲解全局分区索引:
删除之前创建的本地索引:i_id
SQL> drop index i_id ;
Index dropped
SQL> create index i_id_global on test(id) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace USERS,
4 partition p2 values less than (maxvalue) tablespace USERS
5 );
Index created
SQL> alter table test drop partition p3;
Table altered
ORACLE默认不会自动维护全局分区索引,注意看status列,
select INDEX_NAME,PARTITION_NAME,STATUS from user_ind_partitions where index_name='I_ID_GLOBAL';
INDEX_NAME PARTITION_NAME STATUS
------------------------------ ------------------------------ --------
I_ID_GLOBAL P1 USABLE
I_ID_GLOBAL P2 USABLE
SQL> create index i_id_global on test(data) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
ERROR at line 2:
ORA-14038: GLOBAL partitioned index must be prefixed -------------全局索引必须是前缀索引
SQL> create index i_id_global on test(id,data) global ------------------说明,全局索引的索引列必须要以索引分区键做为开头,这样才是前缀索引
2 partition by range(id) -------------------之前一直混淆了索引列和索引分区键的区别。对于本地索引而言,分区键其实都是表的分区键
--------------------而对于全局索引而言,索引分区键可以是任意字段,但是对应的索引列就必须以索引分区键作为引导列。
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
Index created
SQL> create bitmap index i_id_global on test(id) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
create bitmap index i_id_global on test(id) global
partition by range(id)
( partition p1 values less than (2000) tablespace users,
partition p2 values less than (maxvalue) tablespace users
)
ORA-25113: GLOBAL 可能无法与位图索引一起使用 -------------------全局索引无法与位图索引一起使用
三、全局分区索引不能够将其作为整体重建,必须对每个分区重建
SQL> alter index i_id_global rebuild online nologging;
alter index i_id_global rebuild online nologging
ORA-14086: 不能将分区索引作为整体重建
这个时候可以查询dba_ind_partitions,或者user_ind_partitions,找到partition_name,然后对每个分区重建
SQL> select index_name,partition_name from user_ind_partitions where index_name='I_ID_GLOBAL';
INDEX_NAME PARTITION_NAME
------------------------------ ------------------------------
I_ID_GLOBAL P1
I_ID_GLOBAL P2
SQL> alter index i_id_global rebuild partition p1 online nologging;
Index altered
SQL> alter index i_id_global rebuild partition p2 online nologging;
Index altered
四、关于分区索引的几个视图
dba_ind_partitions 描述了每个分区索引的分区情况,以及统计信息
dba_part_indexes 分区索引的概要统计信息,可以得知每个表上有哪些分区索引,分区索引的类型(local/global)
dba_indexes minus dba_part_indexes (mi
时间总归是要打败你我
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
今天必须弄清楚本地索引、全局索引、前缀索引、非前缀索引这些名词间的关系
--------------------------------------------------------------------------------------------------------------------------------------------------------------------
分区索引分为本地(local index)索引和全局索引(global index)。
其中本地索引又可以分为有前缀(prefix)的索引和无前缀(nonprefix)的索引。而全局索引目前只支持有前缀的索引。
B树索引和位图索引都可以分区,但是HASH索引不可以被分区。
位图索引必须是本地索引。下面就介绍本地索引以及全局索引各自的特点来说明区别;
一、本地索引特点:
1. 本地索引一定是分区索引,分区键等同于表的分区键,分区数等同于表的分区数,一句话,本地索引的分区机制和表的分区机制一样。
2. 如果本地索引的索引列以分区键开头,则称为前缀局部索引。
3. 如果本地索引的列不是以分区键开头,或者不包含分区键列,则称为非前缀索引。
4. 前缀和非前缀索引都可以支持索引分区消除,前提是查询的条件中包含索引分区键。
5. 本地索引只支持分区内的唯一性,无法支持表上的唯一性,因此如果要用本地索引去给表做唯一性约束,则约束中必须要包括分区键列。
6. 本地分区索引是对单个分区的,每个分区索引只指向一个表分区,全局索引则不然,一个分区索引能指向n个表分区,
同时,一个表分区,也可能指向n个索引分区,对分区表中的某个分区做truncate或者move,shrink等,可能会影响到n个全局索引分区,正因为这点,本地分区索引具有更高的可用性。
7. 位图索引只能为本地分区索引。
8. 本地索引多应用于数据仓库环境中。
本地索引:创建了一个分区表后,如果需要在表上面创建索引,并且索引的分区机制和表的分区机制一样,那么这样的索引就叫做本地分区索引。(这里的分区机制指的是什么??)
本地索引是由ORACLE自动管理的,它分为有前缀的本地索引和无前缀的本地索引。什么叫有前缀的本地索引?
有前缀的本地索引就是包含了分区键,并且将其作为引导列的索引。什么叫无前缀的本地索引?无前缀的本地索引就是没有将分区键的前导列作为索引的前导列的索引。下面举例说明:
创建分区表:以id作为分区条件
create table test (id number,data varchar2(100))
partition by RANGE (id)
(
partition p1 values less than (1000) tablespace users,
partition p2 values less than (2000) tablespace users,
partition p3 values less than (maxvalue) tablespace users
);
创建本地索引:以分区条件id作为索引分区
create index i_id on test(id) local; 因为id是分区键,所以这样就创建了一个有前缀的本地索引。
select dbms_metadata.get_ddl('INDEX','I_ID','KING') index_name FROM DUAL;
这条语句查询出来的结果和我执行的语句是一样的,并没有看到转换后的分区语句。
创建无前缀的本地分区索引:
create index i_data on test(data) local;因为data不是分区键,所以这样就创建了一个无前缀的本地索引。(也就是说,一张分区表上面建的分区索引,只要不是以表分区条件作为引导列的索引
都是非前缀本地分区索引)
select dbms_metadata.get_ddl('INDEX','I_DATA','KING') index_name FROM DUAL;
INDEX_NAME
------------------------------------------------------------------
CREATE INDEX "KING"."I_DATA" ON "KING"."TEST" ("DATA")
PCTFREE 10 INITRANS
理想状态是应该看到如下语句:
CREATE INDEX "KING"."I_DATA" ON "KING"."TEST" ("DATA") LOCAL
(PARTITION "P1" TABLESPACE "USERS" ,PARTITION "P2" TABLESPACE "USERS" , PARTITION "P3" TABLESPACE "USERS" );
这种情况下方便我们对本地索引进行理解,比如上面我对本地分区索引的分区机制和表分区机制相同的不理解,从上面的语句就可以得到答复。
索引的分区就按照表的分区进行的,
表的分区条件是ID,相对应的ID值都落在了P1,P2,P3这三个分区里面。
当我们再创建DATA这个非前缀本地索引后,其实DATA的范围域就是根据ID的范围域来划分的。比如P1分区对应着ID小于1000的值,那么DATA在P1分区ID值就是对应着小于1000的值。
从user_part_indexes视图也可以证明刚才创建的索引,一个是有前缀的,一个是无前缀的
select index_name,table_name,partitioning_type,locality,ALIGNMENT from user_part_indexes;
index_name table_name partitioning_type locality ALIGNMENT
2 I_ID TEST RANGE LOCAL PREFIXED
3 I_DATA TEST RANGE LOCAL NON_PREFIXED
二、全局索引特点:
1. 全局索引的分区键和分区数和表的分区键和分区数可能都不相同,表和全局索引的分区机制不一样。()
2. 全局索引可以分区,也可以是不分区索引,全局索引必须是前缀索引,即全局索引的索引列必须是以索引分区键作为其前几列。
3. 全局分区索引的索引条目可能指向若干个分区,因此,对于全局分区索引,即使只截断一个分区中的数据,都需要rebulid若干个分区甚至是整个索引。
4. 全局索引多应用于oltp系统中。
5. 全局分区索引只按范围或者散列hash分区,hash分区是10g以后才支持。
6. oracle9i以后对分区表做move或者truncate的时可以用update global indexes语句来同步更新全局分区索引,用消耗一定资源来换取高度的可用性。
7. 表用a列作分区,索引用b做局部分区索引,若where条件中用b来查询,那么oracle会扫描所有的表和索引的分区,成本会比分区更高,此时可以考虑用b做全局分区索引。
另外oracle不会自动的维护全局分区索引,当我们在对表的分区做修改之后,如果执行修改的语句不加上update global indexes的话,那么索引将不可用。
以上面创建的分区表test为例,讲解全局分区索引:
删除之前创建的本地索引:i_id
SQL> drop index i_id ;
Index dropped
SQL> create index i_id_global on test(id) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace USERS,
4 partition p2 values less than (maxvalue) tablespace USERS
5 );
Index created
SQL> alter table test drop partition p3;
Table altered
ORACLE默认不会自动维护全局分区索引,注意看status列,
select INDEX_NAME,PARTITION_NAME,STATUS from user_ind_partitions where index_name='I_ID_GLOBAL';
INDEX_NAME PARTITION_NAME STATUS
------------------------------ ------------------------------ --------
I_ID_GLOBAL P1 USABLE
I_ID_GLOBAL P2 USABLE
SQL> create index i_id_global on test(data) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
ERROR at line 2:
ORA-14038: GLOBAL partitioned index must be prefixed -------------全局索引必须是前缀索引
SQL> create index i_id_global on test(id,data) global ------------------说明,全局索引的索引列必须要以索引分区键做为开头,这样才是前缀索引
2 partition by range(id) -------------------之前一直混淆了索引列和索引分区键的区别。对于本地索引而言,分区键其实都是表的分区键
--------------------而对于全局索引而言,索引分区键可以是任意字段,但是对应的索引列就必须以索引分区键作为引导列。
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
Index created
SQL> create bitmap index i_id_global on test(id) global
2 partition by range(id)
3 ( partition p1 values less than (2000) tablespace users,
4 partition p2 values less than (maxvalue) tablespace users
5 );
create bitmap index i_id_global on test(id) global
partition by range(id)
( partition p1 values less than (2000) tablespace users,
partition p2 values less than (maxvalue) tablespace users
)
ORA-25113: GLOBAL 可能无法与位图索引一起使用 -------------------全局索引无法与位图索引一起使用
三、全局分区索引不能够将其作为整体重建,必须对每个分区重建
SQL> alter index i_id_global rebuild online nologging;
alter index i_id_global rebuild online nologging
ORA-14086: 不能将分区索引作为整体重建
这个时候可以查询dba_ind_partitions,或者user_ind_partitions,找到partition_name,然后对每个分区重建
SQL> select index_name,partition_name from user_ind_partitions where index_name='I_ID_GLOBAL';
INDEX_NAME PARTITION_NAME
------------------------------ ------------------------------
I_ID_GLOBAL P1
I_ID_GLOBAL P2
SQL> alter index i_id_global rebuild partition p1 online nologging;
Index altered
SQL> alter index i_id_global rebuild partition p2 online nologging;
Index altered
四、关于分区索引的几个视图
dba_ind_partitions 描述了每个分区索引的分区情况,以及统计信息
dba_part_indexes 分区索引的概要统计信息,可以得知每个表上有哪些分区索引,分区索引的类型(local/global)
dba_indexes minus dba_part_indexes (mi