联系:手机/微信(+86 17813235971) QQ(107644445)
作者:惜分飞©版权所有[未经本人同意,不得以任何形式转载,否则有进一步追究法律责任的权利.]
当数据库启动的时报undo相关异常,很多情况下我们不得不使用隐含参数来处理(_offline_rollback_segments和_corrupted_rollback_segments),而这个就需要明确异常的undo回滚段名称。在11g中,undo回滚段的名称发生了改变,在11g之前的版本中,回滚段名称是”_SYSTEMn$”之类,而到了11g回滚段的名称变为了”_SYSTEMn_时间戳$”,因为时间戳我们不知道,所以我们不能通过n的值,来确定回滚段的名称,从而也就不能很明确的使用_offline_rollback_segments和_corrupted_rollback_segments来标明异常回滚段。既然回滚段的名称我们可以通过dba_rollback_segs视图来查询,那么我们可以根据这个视图找到对应的回滚段是存储在哪张基表中,下面为dba_rollback_segs视图对应的sql语句
select un.name, decode(un.user#,1,'PUBLIC','SYS'), ts.name, un.us#, f.file#, un.block#, s.iniexts * ts.blocksize, decode(bitand(ts.flags, 3), 1, to_number(NULL), s.extsize * ts.blocksize), s.minexts, s.maxexts, decode(bitand(ts.flags, 3), 1, to_number(NULL), s.extpct), decode(un.status$, 2, 'OFFLINE', 3, 'ONLINE', 4, 'UNDEFINED', 5, 'NEEDS RECOVERY', 6, 'PARTLY AVAILABLE', 'UNDEFINED'), decode(un.inst#, 0, NULL, un.inst#), un.file# from sys.undo$ un, sys.seg$ s, sys.ts$ ts, sys.file$ f where un.status$ != 1 and un.ts# = s.ts# and un.file# = s.file# and un.block# = s.block# and s.type# in (1, 10) and s.ts# = ts.ts# and un.ts# = f.ts# and un.file# = f.relfile#
通过观察,我们知道回滚段信息是存储在sys.undo$中(name字段表示回滚段名称,status$字段表示回滚段状态[1:DELETE,2:OFFLINE,3:ONLINE,4:UNDEFINED,5:NEEDS RECOVERY,6:PARTLY AVAILABLE,其他表示:UNDEFINED]),通过这个视图我们需要找的到status$为5所对应name。
找这些值较简易的方法就是通过dul抽取sys.undo$表中数据,然后在其他库上还原,然后通过sql语句查询
1.设置dul参数(config.txt)
export true,其他参数根据你的实际情况设置
2.填写system表空间对应的数据文件(control.txt)
[oracle@localhost dul]$ more control.txt 0 0 /u01/oradata/first/system01.dbf
3.odu导出数据
DUL> unload table sys.undo$ Unloading table: UNDO$,object ID: 15 Unloading segment,storage(Obj#=15 DataObj#=15 TS#=0 File#=1 Block#=224 Cluster=0) 21 rows unloaded
4.导入至新库
imp chf/xifenfei file=C:\Users\XIFENFEI\Downloads\SYS_UNDO$.dmp FROMUSER=SYS TOUSER=CHF
5.查询需要处理的回滚段
--数据库版本10g及其以上版本 set pagesize 0 feedback off verify off heading off echo off linesize 1000 select WMSYS.WM_CONCAT(name) from UNDO$ where status$=5; --数据库版本9i及其以下版本(自己拼接) set pagesize 0 feedback off verify off heading off echo off select name from UNDO$ where status$=5;
现在已经找出来了需要处理的回滚段,其他数据库恢复步骤与以前数据库相同。
以前的strings命令找出相应回滚段方法测试
在这里可以看出,每个回滚段都有两个(时间戳不一样),在使用的时候,我们就不能确定使用哪个比较合适(因为不知道哪个此时真正的需要的回滚段,而在spfile中同时标明一个回滚段的两个名称,前面一个会被覆盖),由此可以看出,这种方法暂时不太合适(而且在win的系统中很难使用,即使第三方工具可以打开system表空间数据文件,要找到相关内容并非易事,而且也可以有类此linux中现象出现)
_OFFLINE_ROLLBACK_SEGMENTS is a unsupported init.ora parameter which can allow you to cause logical database corruption.
_CORRUPTED_ROLLBACK_SEGMENTS is more dangerous parameter than _OFFLINE_ROLLBACK_SEGMENTS. It basically prevents access to the listed rollback segments headers and assumes all transactions in them are committed. This can very easily cause logical database corruption.