首页 > 代码库 > ORA-01502 oracle数据库 index索引的两种形式

ORA-01502 oracle数据库 index索引的两种形式

    目前的项目中,我们在跑批次的时候,有个SP总是报错 “ORA-01502: index ‘WBILL_102.PK_A_NOTWEB_ACT_PROVINCE_M‘ or partition of such index is in unusable state” ,解决办法就是重建索引,ALTER INDEX IN_AA REBUILD;

    但是解决完之后,在跑批还是报这个错误,这就让我需要考虑,产生这个问题的原因是什么!

我的错误场景还原:

有个表 CUS_ASSET_MONTH,主键是 PK_CUS_ASSET_MONTH ,主键字段是  CUS_CODE,这个表 是分区表 ,分区字段是 DATA_YM

为了使得我们的批次有重跑的功能,所以批次的第一步骤都是 将本月的数据删除:

ALTER TABLE CUS_ASSET_MONTH TRUNCATE PARTITION PA_CUS_ASSET_MONTH_201412;

然后在往表  CUS_ASSET_MONTH 中 insert ,这个时候就会报错 ORA-01502 

场景还原完毕!

问题分析:

CUS_ASSET_MONTH 表的索引  PK_CUS_ASSET_MONTH  是全局索引,

这是创建索引的语句:

alter table  CUS_ASSET_MONTH  add constraint  PK_CUS_ASSET_MONTH  primary key (cus_code) using index ;

这样创建的索引,默认是 GLOBAL INDEX 

在对 表分区进行 TRUNCATE  操作之后,在进行 insert 就会报错 ORA-01502 。

所以我们需要将 索引改成 LOCAL INDEX 

alter table  CUS_ASSET_MONTH  add constraint  PK_CUS_ASSET_MONTH  primary key (cus_code) using index LOCAL ;

但是在执行这句话的时候,又报错了:ORA-14039:分区列必须构成UNIQUE索引的关键字列子集

因为这是创建的是主键索引,主键索引的字段在分区表中必须要包含分区字段,如果这个主键索引必须要有,那么可以将这个主键索引改成一个普通索引。

我的最终解决办法[主键也有,普通索引也有]:

alter table  CUS_ASSET_MONTH  add constraint  PK_CUS_ASSET_MONTH  primary key (cus_code,data_ym) using index LOCAL ;

create index IN_CUS_ASSET_MONTH  ON CUS_ASSET_MONTH(cus_code) local;

这样我在对表分区进行 TRUNCATE 操作之后,在往这个表中插入数据的时候,就不会再报错了!


PS 局部索引LOCAL INDEX 和 全局索引GLOBAL INDEX 的区别 :

局部索引local index
 
1.        局部索引一定是分区索引,分区键等同于表的分区键,分区数等同于表的分区说,一句话,局部索引的分区机制和表的分区机制一样。
2.        如果局部索引的索引列以分区键开头,则称为前缀局部索引。
3.        如果局部索引的列不是以分区键开头,或者不包含分区键列,则称为非前缀索引。
4.        前缀和非前缀索引都可以支持索引分区消除,前提是查询的条件中包含索引分区键。
5.        局部索引只支持分区内的唯一性,无法支持表上的唯一性,因此如果要用局部索引去给表做唯一性约束,则约束中必须要包括分区键列。
6.        局部分区索引是对单个分区的,每个分区索引只指向一个表分区,全局索引则不然,一个分区索引能指向n个表分区,同时,一个表分区,也可能指向n个索引分区,
          对分区表中的某个分区做truncate或者move,shrink等,可能会影响到n个全局索引分区,正因为这点,局部分区索引具有更高的可用性。
7.        位图索引只能为局部分区索引。
8.        局部索引多应用于数据仓库环境中。
 
全局索引global index
 
1.        全局索引的分区键和分区数和表的分区键和分区数可能都不相同,表和全局索引的分区机制不一样。
2.        全局索引可以分区,也可以是不分区索引,全局索引必须是前缀索引,即全局索引的索引列必须是以索引分区键作为其前几列。
3.        全局分区索引的索引条目可能指向若干个分区,因此,对于全局分区索引,即使只动,截断一个分区中的数据,都需要rebulid若干个分区甚
          至是整个索引。
4.        全局索引多应用于oltp系统中。
5.        全局分区索引只按范围或者散列hash分区,hash分区是10g以后才支持。
6.        oracle9i以后对分区表做move或者truncate的时可以用update global indexes语句来同步更新全局分区索引,用消耗一定资源来换取高度的可用性。
7.        表用a列作分区,索引用b做局部分区索引,若where条件中用b来查询,那么oracle会扫描所有的表和索引的分区,成本会比分区更高,此时可以考虑用b做全局分区索引


 
分区索引字典
 
DBA_PART_INDEXES 分区索引的概要统计信息,可以得知每个表上有哪些分区索引,分区索引的类新(local/global,)
Dba_ind_partitions每个分区索引的分区级统计信息
Dba_indexesminusdba_part_indexes,可以得到每个表上有哪些非分区索引
 
索引重建
 
Alter index idx_name rebuild partition index_partition_name [online nologging]
需要对每个分区索引做rebuild,重建的时候可以选择online(不会锁定表),或者nologging建立索引的时候不生成日志,加快速度。
Alter index rebuild idx_name [online nologging]
对非分区索引,只能整个index重建


ORA-01502 oracle数据库 index索引的两种形式