分区表尽量不要建主键

原地址:http://blog.itpub.net/16396821/viewspace-679681/

   今天有个项目要做升级,此项目用的是公司新产品,处于研发阶段,数据库设计文档不全,因为需要手动对比一下升级前后2个版本的ddl,发现新版本的ddl中有些分区表建的是主键。如:create table ALARM_RESULT

(
  ALARM_RESULT_ID   VARCHAR2(50) not null,
  TIME NUMBER(20)
)tablespace test_DEFAULT
partition by range (TIME)
(
  partition W values less than (5)
    tablespace test_DEFAULT,
  partition P_OTHER values less than (MAXVALUE)
    tablespace test_DEFAULT
);

     此问题已反馈研发,建议取消主键,建唯一性本地索引。

分区表尽量不要建主键,因为建主键的同时会建一个唯一性的全局索引,在drop分区表时如果不指定update global indexes则
会使索引失效,导致数据无法入库。
如果非要建主键,要2种方法:
1.应用上drop 分区表时显示指定update global indexes,
2.将主键上的索引建成本地索引
上述2种方法虽然可以实现,但效果都不好。因为当数据量超大时维护索引也是很大的开销。将主键建成本地索引的方法也比较受限。

将主键建成本地索引方法:
建一个包含(分区列、主键列)的唯一性复合索引,然后在这个索引上建主键
create  unique index  idx_ALARM_RESULT_ID on ALARM_RESULT (ALARM_RESULT_ID,TIME) local; #一定要包含分区列否则会报ORA-14039: 分区列必须构成 UNIQUE 索引的关键字列子集
alter table ALARM_RESULT add primary key (ALARM_RESULT_ID,TIME);
测试分区表建主键导致数据无法入库过程:
create table ALARM_RESULT
(
  ALARM_RESULT_ID   VARCHAR2(50) not null,
  TIME NUMBER(20)
)tablespace test_DEFAULT
partition by range (TIME)
(
  partition W values less than (5)
    tablespace test_DEFAULT,
  partition P_OTHER values less than (MAXVALUE)
    tablespace test
);
alter table ALARM_RESULT  add primary key (ALARM_RESULT_ID)   using index   tablespace test_INDEX;

1.查看索引状态

SQL> select table_name,index_name,status from user_indexes;

TABLE_NAME                     INDEX_NAME                     STATUS
------------------------------ ------------------------------ --------
ALARM_RESULT                   SYS_C003358                    VALID

SQL>

2.查看各个分区的数据
SQL> select ALARM_RESULT_ID,time from ALARM_RESULT  PARTITION(w);

ALARM_RESULT_ID                                                     TIME
-------------------------------------------------- ---------------------
1                                                                      1

SQL>
SQL> select ALARM_RESULT_ID,time from ALARM_RESULT  PARTITION(p_other);

ALARM_RESULT_ID                                                     TIME
-------------------------------------------------- ---------------------
2                                                                     10

SQL>


3.drop分区
SQL> ALTER TABLE ALARM_RESULT  DROP PARTITION  w;

Table altered

SQL>

4.查看索引状态
SQL> select table_name,index_name,status from user_indexes;

TABLE_NAME                     INDEX_NAME                     STATUS
------------------------------ ------------------------------ --------
ALARM_RESULT                   SYS_C003358                    UNUSABLE

SQL>

5.向表中插入数据时报错
SQL> insert into ALARM_RESULT(ALARM_RESULT_ID,time)values(3,15);

insert into ALARM_RESULT(ALARM_RESULT_ID,time)values(3,15)

ORA-01502: 索引'owner.SYS_C003358'或这类索引的分区处于不可用状态

SQL>


6.此时如果重建一下索引,索引状态会恢复为可用状态,数据也可插入。
SQL> alter index SYS_C003358 rebuild;

Index altered

SQL> select table_name,index_name,status from user_indexes;

TABLE_NAME                     INDEX_NAME                     STATUS
------------------------------ ------------------------------ --------
ALARM_RESULT                   SYS_C003358                    VALID

SQL> insert into ALARM_RESULT(ALARM_RESULT_ID,time)values(3,15);

1 row inserted

SQL> commit;

Commit complete

SQL>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值