导入/导出是ORACLE幸存的最古老的两个命令行工具其实我从来不认为Exp/Imp是一种好的备份方式正确的说法是Exp/Imp只能是一个好的转储工具特别是在小型数据库的转储表空间的迁移表的抽取检测逻辑和物理沖突等中有不小的功劳当然我们也可以把它作为小型数据库的物理备份后的一个逻辑辅助备份也是不错的建议对于越来越大的数据库特别是TB级数据库和越来越多数据仓库的出现EXP/IMP越来越力不从心了这个时候数据库的备份都转向了RMAN和第三方工具下面说明一下EXP/IMP的使用
如何使exp的帮助以不同的字符集显示set nls_lang=simplified chinese_chinazhsgbk通过设置环境变量可以让exp的帮助以中文显示如果set nls_lang=American_america字符集那么帮助就是英文的了
EXP的所有参数(括号中为参数的默认值)
USERID 用户名/口令 如 USERID=duanl/duanl
FULL 导出整个数据库 (N)
BUFFER 数据缓沖区的大小
OWNER 所有者用户名列表你希望导出哪个用户的对象就用owner=username
FILE 输出文件 (EXPDATDMP)
TABLES 表名列表 指定导出的table名称如TABLES=tabletable
COMPRESS 导入一个extent (Y)
RECORDLENGTH IO 记录的长度
GRANTS 导出权限 (Y)
INCTYPE 增量导出类型
INDEXES 导出索引 (Y)
RECORD 跟蹤增量导出 (Y)
ROWS 导出数据行 (Y)
PARFILE 参数文件名如果你exp的参数很多可以存成参数文件
CONSTRAINTS 导出约束 (Y)
CONSISTENT 交叉表一致性
LOG 屏幕输出的日志文件
STATISTICS 分析对象 (ESTIMATE)
DIRECT 直接路径 (N)
TRIGGERS 导出触发器 (Y)
FEEDBACK 显示每 x 行 () 的进度
FILESIZE 各转储文件的最大尺寸
QUERY 选定导出表子集的子句
下列关键字仅用于可传输的表空间
TRANSPORT_TABLESPACE 导出可传输的表空间元数据 (N)
TABLESPACES 将传输的表空间列表
IMP的所有参数(括号中为参数的默认值)
USERID 用户名/口令
FULL 导入整个文件 (N)
BUFFER 数据缓沖区大小
FROMUSER 所有人用户名列表
FILE 输入文件 (EXPDATDMP)
TOUSER 用户名列表
SHOW 只列出文件内容 (N)
TABLES 表名列表
IGNORE 忽略创建错误 (N)
RECORDLENGTH IO 记录的长度
GRANTS 导入权限 (Y)
INCTYPE 增量导入类型
INDEXES 导入索引 (Y)
COMMIT 提交数组插入 (N)
ROWS 导入数据行 (Y)
PARFILE 参数文件名
LOG 屏幕输出的日志文件
CONSTRAINTS 导入限制 (Y)
DESTROY 覆盖表空间数据文件 (N)
INDEXFILE 将表/索引信息写入指定的文件
SKIP_UNUSABLE_INDEXES 跳过不可用索引的维护 (N)
ANALYZE 执行转储文件中的 ANALYZE 语句 (Y)
FEEDBACK 显示每 x 行 () 的进度
TOID_NOVALIDATE 跳过指定类型 id 的校验
FILESIZE 各转储文件的最大尺寸
RECALCULATE_STATISTICS 重新计算统计值 (N)
下列关键字仅用于可传输的表空间
TRANSPORT_TABLESPACE 导入可传输的表空间元数据 (N)
TABLESPACES 将要传输到数据库的表空间
DATAFILES 将要传输到数据库的数据文件
TTS_OWNERS 拥有可传输表空间集中数据的用户
关于增量参数的说明exp/imp的增量并不是真正意义上的增量所以最好不要使用
使用方法
Exp parameter_name=value or Exp parameter_name=(valuevalue……)
只要输入参数help=y就可以看到所有帮助
EXP常用选项
FULL 这个用于导出整个数据库在ROWS=N一起使用时可以导出整个数据库的结构例如
exp userid=test/test file=/db_strdmp log=/db_strlog full=y rows=n compress=y direct=y
注在oracleg中userid这个是可以不用写的直接写 用户名/密码 就可以了像这样
exp test/test file=/db_strdmp log=/db_strlog full=y rows=n compress=y direct=y
OWNER和TABLE 这两个选项用于定义EXP的对象OWNER定义导出指定用户的对象;TABLE指定EXP的table名称例如
exp userid=test/test file=/db_strdmp log=/db_strlog owner=duanl
注:有的时候我们可能会遇到只导出单表的情况下面这种方法可以帮你解决问题
exp userid=test/test file=/db_strdmp log=/db_strlog table=nc_datafi_arap
BUFFER和FEEDBACK 在导出比较多的数据时我会考虑设置这两个参数例如
exp userid=test/test file=yw_dmp log=yw__log feedback= buffer= tables=WOOK_YT
FILE和LOG 这两个参数分别指定备份的DMP名称和LOG名称包括文件名和目录例子见上面
COMPRESS 参数不压缩导出数据的内容用来控制导出对象的storage语句如何产生默认值为Y使用默认值对象的存储语句的init extent等于当前导出对象的extent的总和推荐使用COMPRESS=N
FILESIZE该选项在i中可用 如果导出的dmp文件过大时最好使用FILESIZE参数限制文件大小不要超过G如
exp userid=duanl/duanl file=fffff filesize=G owner=scott
这样将创建fdmp fdmp等一系列文件每个大小都为G如果导出的总量小于G
EXP不必创建fbmp
IMP常用选项
FROMUSER和TOUSER 使用它们实现将数据从一个SCHEMA中导入到另外一个SCHEMA中例如假设我们做exp时导出的为test的对象现在我们想把对象导入用户
imp userid=test/test file=expdatdmp fromuser=test touser=test
IGNOREGRANTS和INDEXES 其中IGNORE参数将忽略表的存在继续导入这个对于需要调整表的存储参数时很有用我们可以先根据实际情况用合理的存储参数建好表然后直接导入数据而GRANTS和INDEXES则表示是否导入授权和索引如果想使用新的存储参数重建索引或者为了加快到入速度我们可以考虑将INDEXES设为N而GRANTS一般都是Y例如imp userid=test/test file=expdatdmp fromuser=test touser=test indexes=N
表空间传输
表空间传输是i新增加的一种快速在数据库间移动数据的一种办法是把一个数据库上的格式数据文件附加到另外一个数据库中而不是把数据导出成Dmp文件这在有些时候是非常管用的因为传输表空间移动数据就象复制文件一样快
关于传输表空间有一些规则即
·源数据库和目标数据库必须运行在相同的硬件平台上
·源数据库与目标数据库必须使用相同的字符集
·源数据库与目标数据库一定要有相同大小的数据块
·目标数据库不能有与迁移表空间同名的表空间
·SYS的对象不能迁移
·必须传输自包含的对象集
·有一些对象如物化视图基于函数的索引等不能被传输
可以用以下的方法来检测一个表空间或一套表空间是否符合传输标准
exec sysdbms_ttstransport_set_check(‘tablespace_name’true);
select * from systransport_set_violation;
如果没有行选择表示该表空间只包含表数据并且是自包含的对于有些非自包含的表空间如数据表空间和索引表空间可以一起传输
以下为简要使用步骤如果想参考详细使用方法也可以参考ORACLE联机帮助
设置表空间为只读(假定表空间名字为APP_Data 和APP_Index)
alter tablespace app_data read only;
alter tablespace app_index read only;
发出EXP命令
SQL>host exp userid=”””sys/password as sysdba”””
transport_tablespace=y tablespace=(app_data app_index)
以上需要注意的是
·为了在SQL中执行EXPUSERID必须用三个引号在UNIX中也必须注意避免“/”的使用
·在和以后必须使用sysdba才能操作
·这个命令在SQL中必须放置在一行(这里是因为显示问题放在了两行)
拷贝数据文件到另一个地点即目标数据库
可以是cp(unix)或copy(windows)或通过ftp传输文件(一定要在bin方式)
把本地的表空间设置为读写
在目标数据库附加该数据文件
imp file=expdatdmp userid=”””sys/password as sysdba””” transport_tablespace=y “datafile=(c:tempapp_datac:tempapp_index)”
设置目标数据库表空间为读写
alter tablespace app_data read write;
alter tablespace app_index read write;
优化EXP/IMP的方法
当需要exp/imp的数据量比较大时这个过程需要的时间是比较长的我们可以用一些方法来优化exp/imp的操作
exp:使用直接路径 direct=y
oracle会避开sql语句处理引擎直接从数据库文件中读取数据然后写入导出文件
可以在导出日志中观察到: exp: table xxx will be exported in conventional path
如果没有使用直接路径必须保证buffer参数的值足够大
有一些参数于direct=y不兼容无法用直接路径导出可移动的tablespace或者用query参数导出数据库子集
当导入导出的数据库运行在不同的os下时必须保证recordlength参数的值一致
imp:通过以下几个途径优化
避免磁盘排序
将sort_area_size设置为一个较大的值比如M
避免日志切换等待
增加重做日志组的数量增大日志文件大小
优化日志缓沖区
比如将log_buffer容量扩大倍(最大不要超过M)
使用阵列插入与提交
commit = y
注意:阵列方式不能处理包含LOB和LONG类型的表对于这样的table如果使用commit = y每插入一行就会执行一次提交
使用NOLOGGING方式减小重做日志大小
在导入时指定参数indexes=n只导入数据而忽略index在导完数据后在通过脚本创建index指定 NOLOGGING选项
导出/导入与字符集
进行数据的导入导出时我们要注意关于字符集的问题在EXP/IMP过程中我们需要注意四个字符集的参数导出端的客户端字符集导出端数据库字符集导入端的客户端字符集导入端数据库字符集
我们首先需要查看这四个字符集参数
查看数据库的字符集的信息
SQL> select * from nls_database_parameters;
PARAMETER VALUE
NLS_LANGUAGE AMERICAN
NLS_TERRITORY AMERICA
NLS_CURRENCY $
NLS_ISO_CURRENCY AMERICA
NLS_NUMERIC_CHARACTERS
NLS_CHARACTERSET ZHSGBK
NLS_CALENDAR GREGORIAN
NLS_DATE_FORMAT DDMONRR
NLS_DATE_LANGUAGE AMERICAN
NLS_SORT BINARY
NLS_TIME_FORMAT HHMISSXFF AM
NLS_TIMESTAMP_FORMAT DDMONRR HHMISSXFF AM
NLS_TIME_TZ_FORMAT HHMISSXFF AM TZH:TZM
NLS_TIMESTAMP_TZ_FORMAT DDMONRR HHMISSXFF AM TZH:TZM
NLS_DUAL_CURRENCY $
NLS_COMP BINARY
NLS_NCHAR_CHARACTERSET ZHSGBK
NLS_RDBMS_VERSION
NLS_CHARACTERSETZHSGBK是当前数据库的字符集
我们再来查看客户端的字符集信息
客户端字符集的参数NLS_LANG=_< territory >
language指定oracle消息使用的语言日期中日和月的显示
Territory指定货币和数字的格式地区和计算星期及日期的习惯
Characterset控制客户端应用程序使用的字符集通常设置或等于客户端的代码页或者对于unicode应用设为UTF
在windows中查询和修改NLS_LANG可在注册表中进行
HKEY_LOCAL_MACHINESOFTWAREOracleHOMExx
xx指存在多个Oracle_HOME时的系统编号
在unix中
$ env|grep NLS_LANG
NLS_LANG=simplified chinese_chinaZHSGBK
修改可用
$ export NLS_LANG=AMERICAN_AMERICAUTF
通常在导出时最好把客户端字符集设置得和数据库端相同当进行数据导入时主要有以下两种情况
() 源数据库和目标数据库具有相同的字符集设置
这时只需设置导出和导入端的客户端NLS_LANG等于数据库字符集即可
() 源数据库和目标数据库字符集不同
先将导出端客户端的NLS_LANG设置成和导出端的数据库字符集一致导出数据然后将导入端客户端的NLS_LANG设置成和导出端一致导入数据这样转换只发生在数据库端而且只发生一次
这种情况下只有当导入端数据库字符集为导出端数据库字符集的严格超集时数据才能完全导成功否则可能会有数据不一致或乱码出现
不同版本的EXP/IMP问题
一般来说从低版本导入到高版本问题不大麻烦的是将高版本的数据导入到低版本中在Oraclei之前不同版本Oracle之间的EXP/IMP可以通过下面的方法来解决
在高版本数据库上运行底版本的catexpsql;
使用低版本的EXP来导出高版本的数据;
使用低版本的IMP将数据库导入到低版本数据库中;
在高版本数据库上重新运行高版本的catexpsql脚本
但在i中上面的方法并不能解决问题如果直接使用低版本EXP/IMP会出现如下错误
EXP: orACLE error %lu encountered
orA: invalid column name
这已经是一个公布的BUG需要等到Oracle才能解决BUG号为你可以到METALINK上去查看有关此BUG的详细信息
BUG归BUG我们的工作还是要做在没有Oracle的支持之前我们就自己解决在Oraclei中执行下面的SQL重建exurls视图即可
Create or REPLACE view exurls
(objownobjnampolicypolownpolschpolfunstmtschkoptenabledspolicy)
AS select uname oname rpname rpfschma rppname rpfname
decode(bitand(rstmt_type) Select)
|| decode(bitand(rstmt_type) Insert)
|| decode(bitand(rstmt_type) Update)
|| decode(bitand(rstmt_type) Delete)
rcheck_opt renable_flag
DECODE(BITAND(rstmt_type ) )
from user$ u obj$ o rls$ r
where uuser# = oowner#
and robj# = oobj#
and (uid = or
uid = oowner# or
exists ( select * from session_roles where role=Select_CATALOG_ROLE)
)
/
grant select on sysexurls to public;
/
可以跨版本的使用EXP/IMP但必须正确地使用EXP和IMP的版本
总是使用IMP的版本匹配数据库的版本如要导入到中使用的IMP工具
总是使用EXP的版本匹配两个数据库中最低的版本如从往中导入则使用版本的EXP工具