发现漏了一个细节,就是表、LOB对象移动后索引的状态。

1.创建测试表

CREATE TABLE tbl_able(id NUMBER PRIMARY KEY, con CLOB);

该表有一个CLOB的大对象。

SELECT * FROM user_tables WHERE table_name='TBL_ABLE';
SELECT * FROM user_indexes WHERE table_name='TBL_ABLE';

TBL_ABLE表存在于用户默认表空间TEST_TBS中。
user_indexes中保存两条TBL_ABLE的记录,一条是LOB类型的SYS_IL0000124396C00002$$,一条是主键索引SYS_C0055438,都存储于TEST_TBS表空间。

2.将表移至USERS表空间

ALTER TABLE tbl_able MOVE TABLESPACE USERS;

此时查看user_indexes,LOB对象SYS_IL0000124401C00002$$的状态仍是VALID,但主键索引SYS_C0055439的状态变为了UNUSABLE,且两者所处的表空间仍是TEST_TBS。
此时INSERT操作会提示:

ORA-01502:索引’SYS_C0055439’或这类索引的分区处于不可用状态

3.移动索引对象

ALTER INDEX SYS_C0055439 REBUILD TABLESPACE USERS;

将主键索引移至USERS表空间,且此时状态是VALID,LOB对象SYS_IL0000124401C00002$$没有改变,状态仍是VALID。

ALTER TABLE TBL_ABLE MOVE LOB(con) store as (TABLESPACE USERS);

将LOB对象SYS_IL0000124401C00002$$移至USERS表空间,此时LOB对象状态是VALID,但主键索引状态现在却是UNUSABLE了。

这里我使用了11.2.0.1和11.2.0.4两个库版本测试,发现在11.2.0.4中执行alter table move或alter table move lob时将主键索引状态置为UNUSABLE,alert日志中会有记录:

Some indexes or index [sub]partitions of table XXX.TBL_ABLE have been marked unusable

标记了索引错误,这和INSERT操作报错ORA-01502的描述不谋而合,但11.2.0.1版本的alert日志中没有任何标记。

4.让索引重新生效

ALTER INDEX SYS_C0055439 REBUILD;

重建索引,此时才能改为VALID状态。

总结:
对于对象移动表空间的操作,要关注索引等附属信息的有效性,对于LOB对象的移动,也可能会导致索引状态的失效。