ITPub博客

首页 > Linux操作系统 > Linux操作系统 > EXPDP为什么无法导出只读表空间

EXPDP为什么无法导出只读表空间

原创 Linux操作系统 作者:fengjin821 时间:2009-06-07 19:17:39 0 删除 编辑

想用EXPDPD做数据迁移
但是TRANSPORT表空间必须更改变空间为只读表空间!
现在报错,错误如下
[oracle@localhost ~]$ expdp napo/napo directory=napo dumpfile=transport.dmp transport_tablespaces=napo

Export: Release 10.2.0.1.0 - Production on Saturday, 06 June, 2009 16:10:47

Copyright (c) 2003, 2005, Oracle.  All rights reserved.

Connected to: Oracle Database 10g Enterprise Edition Release 10.2.0.1.0 - Production
With the Partitioning, OLAP and Data Mining options
ORA-31626: job does not exist
ORA-31633: unable to create master table "NAPO.SYS_EXPORT_TRANSPORTABLE_05"
ORA-06512: at "SYS.DBMS_SYS_ERROR", line 95
ORA-06512: at "SYS.KUPV$FT", line 863
ORA-01647: tablespace 'NAPO' is read only, cannot allocate space in it

表空间状态:
SQL> select status  from dba_tablespaces where tablespace_name='NAPO';

STATUS
---------
READ ONLY

要修改表空间为READ ONLY
NAPO用户的默认表空间就是NAPO表空间~
NAPO表空间是SYS创建的

这是一个有趣的error。
根本原因是做expdp的时候,使用expdp的用户会建立一个master table。
refer to“needed to create the so-called Master Table that is used by the DataPump job”。

而这个master table当然是create在这个用户的默认表空间里的,当expdp结束或者意外终止时,自动删除这个表。
所以当我们在做expdp时,使用这个用户连进去看,可以看到中途产生了这个master table:
SQL> select * from tab;

no rows selected


SQL> /

TNAME                                                                                      TABTYPE                CLUSTERID
------------------------------------------------------------------------------------------ --------------------- ----------
SYS_EXPORT_TRANSPORTABLE_01                                                                TABLE

SQL> /

no rows selected

所以,当你把这个用户的默认表空间readonly后,再用这个用户expdp,当然就建立不上这个master table了。
所以,只能换个用户expdp。

表空间需要TTS-执行完整的或部分相关性检查,所以不能把表空间修改为read write,必须是read only

先用TRANSPORT_SET_CHECK检验空间的是否为自包含表空间,且不能为system和sysaux,sys用户所在的表空间就可以搬移了

TRANSPORT_TABLESPACE 导出可传输的表空间元数据 (N)

通过这个选项,我们可以对一组自包含、只读的表空间只导出元数据,然后在操作系统层将这些表空间的数据文件拷贝至目标平台,并将元数据导入数据字典(这个过程称为插入,plugging),即完成迁移。


对于可传输表空间有一个重要概念:自包含(Self-Contained)。
在表空间传输的中,要求表空间集为自包含的,自包含表示用于传输的内部表空间集没有引用指向外部表空间集。自包含分为两种:一般自包含表空间集和完全(严格)自包含表空间集。


常见的以下情况是违反自包含原则的:


 索引在内部表空间集,而表在外部表空间集(相反地,如果表在内部表空间集,而索引在外部表空间集,则不违反自包含原则)。


 分区表一部分区在内部表空间集,一部分在外部表空间集(对于分区表,要么全部包含在内部表空间集中,要么全不包含)。


 如果在传输表空间时同时传输约束,则对于引用完整性约束,约束指向的表在外部表空间集,则违反自包含约束;如果不传输约束,则与约束指向无关。


 表在内部表空间集,而lob列在外部表空间集,则违反自包含约束。


通常可以通过系统包DBMS_TTS来检查表空间是否自包含,验证可以以两种方式执行:非严格方式和严格方式。


以下是一个简单的验证过程,假定在eygle表空间存在一个表eygle,其上存在索引存储在USERS表空间:

SQL> create table eygle as select rownum id ,username from dba_users;
Table created.

SQL> create index ind_id on eygle(id) tablespace users;
Index created.

以SYS用户执行非严格自包含检查(full_check=false):

SQL> connect / as sysdba
Connected.
SQL> exec dbms_tts.transport_set_check('EYGLE', TRUE);
PL/SQL procedure successfully completed.

SQL> SELECT * FROM TRANSPORT_SET_VIOLATIONS;
no rows selected

执行严格自包含检查(full_check=true):

SQL> exec dbms_tts.transport_set_check('EYGLE', TRUE, True);
PL/SQL procedure successfully completed.

SQL> SELECT * FROM TRANSPORT_SET_VIOLATIONS;
VIOLATIONS
--------------------------------------------------------------------------------------
Index EYGLE.IND_ID in tablespace USERS points to table EYGLE.EYGLE in tablespace EYGLE

反过来对于USERS表空间来说,非严格检查也是无法通过的:

SQL> exec dbms_tts.transport_set_check('USERS', TRUE);
PL/SQL procedure successfully completed.

SQL> SELECT * FROM TRANSPORT_SET_VIOLATIONS;
VIOLATIONS
----------------------------------------------------------------------------------------
Index EYGLE.IND_ID in tablespace USERS points to table EYGLE.EYGLE in tablespace EYGLE

但是可以对多个表空间同时传输,则一些自包含问题就可以得到解决:

SQL> exec dbms_tts.transport_set_check('USERS,EYGLE', TRUE, True);
PL/SQL procedure successfully completed.

SQL> SELECT * FROM TRANSPORT_SET_VIOLATIONS;
no rows selected

表空间自包含确认之后,进行表空间传输就很方便了

 

一般包含如下几个步骤。


(1) 将表空间设置为只读:
alter tablespace users read only;


(2) 导出表空间。在操作系统提示符下执行:
exp username/passwd tablespaces=users transport_tablespace=y file=exp_users.dmp
此处的导出文件只包含元数据,所以导出文件很小,导出速度也会很快。


(3) 转移。将导出的元数据文件(此处是exp_users.dmp)和传输表空间的数据文件(此处是users表空间的数据文件user01.dbf)转移至目标主机(转移过程如果使用FTP方式,应该注意使用二进制方式)。


(4) 传输。在目标数据库将表空间插入到数据库中,完成表空间传输。在操作系统命令提示符下执行下面的语句:
imp username/passwd tablespaces=users transport_tablespace=y file=exp_users.dmp datafiles='users01.dbf'

了解了Oracle的可传输表空间技术后,来看一下example表空间的插入,以下脚本仍然来自mkplug.sql脚本:

--
-- Importing the metadata and plugging in the tablespace at the same
-- time, using the restored database file
--
DEFINE imp_logfile = &log_path.tts_example_imp.log

-- When importing use filename got after restore is finished
host imp "'sys/&&password_sys AS SYSDBA'" transport_tablespace=y file=&imp_file log=&imp_logfile datafiles='&datafile' tablespaces=EXAMPLE tts_owners=hr,oe,pm,ix,sh


完成plugging之后,这个表空间就被包含在了新建的数据库之中。

 

 

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/13165828/viewspace-605392/,如需转载,请注明出处,否则将追究法律责任。

上一篇: RAC ORA_12545错误
请登录后发表评论 登录
全部评论

注册时间:2009-04-29

  • 博文量
    191
  • 访问量
    505316