用expdp导出生产库数据到测试库,执行impdp的时候报了ORA-02298错误,提示生效TBL_B表的外键约束FK_B_ID的时候出错,
data:image/s3,"s3://crabby-images/55655/55655eb41c246cbf978147526b6638419f18b5a0" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
看看ORA-02298的错误描述,因为存在独立的字节点记录,导致生效约束操作报错,
data:image/s3,"s3://crabby-images/45b2c/45b2c618e182490c3a4e7f5fa1e5021ee7e76c9b" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
通俗一些,就是子表外键对应的主表主键/唯一约束键值不存在,所以此时无法生效外键约束。
方案1:
既然错误提示子表存在一些主表无记录的外键值,那么只要找出这些不符合主外键关系的子表记录,并且删除这些,保证子表中的外键记录,主表中均有对应的记录。
创建测试表和相应数据,
data:image/s3,"s3://crabby-images/1c407/1c407b7c8863195efb64837586c76bb29dae835b" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
主表不存在id=2这条记录,但子表中存在外键字段id_a=2的这条记录,只是由于disable了约束所以才可以insert,但实际此时是无法enable约束,这和上面执行impdp的效果相同,
data:image/s3,"s3://crabby-images/32f16/32f16d834be4ed21589a2e4d8e30f5786fe5b688" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
使用如下SQL,可以找出子表TBL_B中外键字段id_a的值未在主表TBL_A中有定义的记录,并且删除,
data:image/s3,"s3://crabby-images/e515c/e515cffcb70de8ad53c26dc13b118ad4ecbb6c40" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
此时就可以正常enable约束。
使用如下SQL,可以根据子表名称和子表外键约束名称,自动拼接出需要删除子表非法数据的SQL语句,复制出来继续执行就行,
SELECT ' delete from ' || a.table_name || ' a where not exists ( select 1 from ' || c_pk.table_name || ' b where b.' || b.column_name || '=a.' || a.column_name ||');' FROM user_cons_columns a JOIN user_constraints c ON a.constraint_name = c.constraint_name JOIN user_constraints c_pk ON c.r_constraint_name = c_pk.constraint_name JOIN user_cons_columns b ON c_pk.constraint_name = b.constraint_name WHERE c.constraint_type = 'R' AND upper(a.table_name) = upper('&Table_Name') AND upper(a.constraint_name) = upper('&FK_NAME');
可以从我的GitHub上下载这一个SQL脚本,
https://github.com/bisal-liu/oracle/blob/832c9c34c068981405a68bae55de885d78cf7bca/solve_illegal_constraint_data
方案2:
出现错误的根本原因,是因为expdp导出的过程中,对于数据表是有DML操作的,即执行expdp指令导出的数据并不能确保属于同一个事务,要从根本解决这问题,就需要确保执行expdp的操作对应的数据属于同一个事务。
exp下可以使用consistent参数,默认值是N,
CONSISTENT cross-table consistency(N)
使用consistent=y,则会设置set transaction read only,即使用了只读事务机制,保证exp导出数据属于一个事务了,
data:image/s3,"s3://crabby-images/5bd5c/5bd5c55f5b803ac1f38c8377a85a056563df85f9" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
但其有一些弊端,例如由于需要读取回滚段中未提交的事务数据,因此exp表会变慢,同时官方文档列出了一些使用consistent=y的适用场景以及注意事项,
data:image/s3,"s3://crabby-images/66cc0/66cc0ec13f89075c7b90ef04ecc2d612b436a8d5" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
expdp下可以使用flashback_scn和flashback_time参数,和闪回表类似,支持设置SCN和TIME两种,
data:image/s3,"s3://crabby-images/64e3d/64e3ddc79558a859c1fcfc888a5ab12293ac44ea" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
FLASHBACK_SCN介绍,
data:image/s3,"s3://crabby-images/c6d14/c6d141fad70145e82462a8bfa8fcdb63b71895cb" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
FLASHBACK_TIME介绍,
data:image/s3,"s3://crabby-images/5a4b9/5a4b948dafb1bd7688015ed4c0a7aade357dbfef" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
data:image/s3,"s3://crabby-images/54ff5/54ff5b9fc51dc22f3c72a42e10135ce838179e8b" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"
总结:
解决ORA-02289错误,要理解其本质,即子表外键值存在不属于主表主键/唯一约束键的情况。
一种方法是手工删除子表中存在的非法数据,保证主子表关系正确。
一种方法是保证导出的时候就要求数据属于同一事物,不受其他事务的影响,此时exp有consistent参数,expdp有flashback_scn和flashback_time参数可以支持此操作,而且需要清楚用这些参数的原理、弊端,以及适用场景。
如果您觉得此篇文章对您有帮助,欢迎关注微信公众号:bisal的个人杂货铺,您的支持是对我最大的鼓励!共同学习,共同进步:)
data:image/s3,"s3://crabby-images/f2ff7/f2ff772247511c7f6e1e66fc4897cf1e1290ca29" alt="解决导入过程中出现的ORA-02289错误 解决导入过程中出现的ORA-02289错误"