分区表创建后一般需要创建分区索引。分区索引分为本地(local index)索引和全局索引(global index)。
其中本地索引又可以分为有前缀(prefix)的索引和无前缀(nonprefix)的索引。而全局索引目前只支持有前缀的索引。B树索引和位图索引都可以分区,但是HASH索引不可以被分区。位图索引必须是本地索引。
下面我们就来一一介绍下:
a.local index 结构:
b.local index 特点:
1.本地索引一定是分区索引,分区键等同于表的分区键,分区数等同于表的分区数,一句话,本地索引的分区机制和表的分区机制一样。
2.如果本地索引的索引列以分区键开头,则称为前缀局部索引。
3.如果本地索引的列不是以分区键开头,或者不包含分区键列,则称为非前缀索引。
4.前缀和非前缀索引都可以支持索引分区消除,前提是查询的条件中包含索引分区键。
5.本地索引只支持分区内的唯一性,无法支持表上的唯一性,因此如果要用本地索引去给表做唯一性约束,则约束中必须要包括分区键列。
6.本地分区索引是对单个分区的,每个分区索引只指向一个表分区,全局索引则不然,一个分区索引能指向n个表分区,同时,一个表分区,也可能指向n个索引分区,对分区表中的某个分区做truncate或者move,shrink等,可能会影响到n个全局索引分区,正因为这点,本地分区索引具有更高的可用性。
7.位图索引只能为本地分区索引。
8.本地索引多应用于数据仓库环境中。
创建语法:CREATE INDEX INX_TAB_PARTITION_COL1 ON TABLE_PARTITION(COL1) LOCAL;
示例:
create table test (id number,data varchar2(100))
partition by RANGE (id)
(
partition p1 values less than (1000) tablespace p1,
partition p2 values less than (2000) tablespace p2,
partition p3 values less than (maxvalue) tablespace p3
);
create index i_id on test(id) local; 因为id是分区键,所以这样就创建了一个有前缀的本地索引。
create index i_data on test(data) local;因为data不是分区键,所以这样就创建了一个无前缀的本地索引。
本地索引:创建了一个分区表后,如果需要在表上面创建索引,并且索引的分区机制和表的分区机制一样,那么这样的索引就叫做本地分区索引。本地索引是由ORACLE自动管理的,它分为有前缀的本地索引和无前缀的本地索引。什么叫有前缀的本地索引?有前缀的本地索引就是包含了分区键,并且将其作为引导列的索引。什么叫无前缀的本地索引?无前缀的本地索引就是没有将分区键的前导列作为索引的前导列的索引。
1. 全局索引的分区键和分区数和表的分区键和分区数可能都不相同,表和全局索引的分区机制不一样。
2. 全局索引可以分区,也可以是不分区索引,全局索引必须是前缀索引,即全局索引的索引列必须是以索引分区键作为其前几列。
3. 全局索引可以依附于分区表;也可以依附于非分区表。
4. 全局分区索引的索引条目可能指向若干个分区,因此,对于全局分区索引,即使只截断一个分区中的数据,都需要rebulid若干个分区甚至是整个索引。
5. 全局索引多应用于oltp系统中。
6. 全局分区索引只按范围或者散列分区,hash分区是10g以后才支持。
7. oracle9i以后对分区表做move或者truncate的时可以用update global indexes语句来同步更新全局分区索引,用消耗一定资源来换取高度的可用性。
8. 表用a列作分区,索引用b做局部分区索引,若where条件中用b来查询,那么oracle会扫描所有的表和索引的分区,成本会比分区更高,此时可以考虑用b做全局分区索引。
DBA_PART_INDEXES 分区索引的概要统计信息,可以得知每个表上有哪些分区索引,分区索引的类型(local/global)
Dba_ind_partitions 每个分区索引的分区级统计信息
Dba_indexes/dba_part_indexes 可以得到每个表上有哪些非分区索引
Alter index idx_name rebuild partition index_partition_name [online nologging]
需要对每个分区索引做rebuild,重建的时候可以选择online(不会锁定表),或者nologging建立索引的时候不生成日志,加快速度。
Alter index rebuild idx_name [online nologging]
对非分区索引,只能整个index重建
分区索引示例:
--1、建分区表 CREATE TABLE P_TAB( C1 INT, C2 VARCHAR2(16), C3 VARCHAR2(64), C4 INT , CONSTRAINT PK_PT PRIMARY KEY (C1) ) PARTITION BY RANGE(C1)( PARTITION P1 VALUES LESS THAN (10000000), PARTITION P2 VALUES LESS THAN (20000000), PARTITION P3 VALUES LESS THAN (30000000), PARTITION P4 VALUES LESS THAN (MAXVALUE) ); --2、建全局分区索引 CREATE INDEX IDX_PT_C4 ON P_TAB(C4) GLOBAL PARTITION BY RANGE(C4) ( PARTITION IP1 VALUES LESS THAN(10000), PARTITION IP2 VALUES LESS THAN(20000), PARTITION IP3 VALUES LESS THAN(MAXVALUE) ); --3、建本地分区索引 CREATE INDEX IDX_PT_C2 ON P_TAB(C2) LOCAL (PARTITION P1,PARTITION P2,PARTITION P3,PARTITION P4); --4、建全局分区索引(与分区表分区规则相同的列上) CREATE INDEX IDX_PT_C1 ON P_TAB(C1) GLOBAL PARTITION BY RANGE (C1) ( PARTITION IP01 VALUES LESS THAN (10000000), PARTITION IP02 VALUES LESS THAN (20000000), PARTITION IP03 VALUES LESS THAN (30000000), PARTITION IP04 VALUES LESS THAN (MAXVALUE) ); --5、分区索引数据字典查看 SELECT * FROM USER_IND_PARTITIONS; SELECT * FROM USER_PART_INDEXES;
一般使用LOCAL索引较为方便,而且维护代价较低,并且LOCAL索引是在分区的基础上去创建索引,类似于在一个子表内部去创建索引,这样开销主要是区分分区上,很规范的管理起来,在OLAP系统中应用很广泛;而相对的GLOBAL索引是全局类型的索引,根据实际情况可以调整分区的类别,而并非按照分区结构一一定义,相对维护代价较高一些,在OLTP环境用得相对较多,这里所谓OLTP和OLAP也是相对的,不是特殊的项目,没有绝对的划分概念,在应用过程中依据实际情况而定,来提高整体的运行性能。