使用ODU恢复Truncate表
ODUmanual ODU3月 15th, 2009意外Truncate表的事情时有发生,ODU提供了方便的恢复Truncate表的功能。被Truncate的表,只要原来的空间没有被重用(即数据被覆盖),则数据都是可以恢复的。
如果发现一个表被意外地Truncate,而需要马上恢复。首先要做的就是关闭数据库,或者OFFLINE那个表所在的表空间,或者关闭所有应用。目的只有一个,确保空间不会被重用,数据不会被覆盖。
下面举例说明如何用ODU恢复被Truncate掉的表。
1. 建立两个测试的表T1和T2,这两个表的数据完全一样。建两个数据完全一样的表的目的在于方便在恢复后对比数据。
SQL> connect test/test
已连接。SQL> create table t1 as select * from dba_objects;
表已创建。
SQL> create table t2 as select * from t1;
表已创建。
SQL> truncate table t1;
表已截掉。
2. 我们OFFLINE掉T1表的表空间(实际上在实际的系统中,如果有比较多的活动,则表空间不容易被OFFLINE下来)。然后做一个Checkpoint,让ODU能够读到最新的数据字典数据。
SQL> select tablespace_name from user_tables where table_name=’T1′;
TABLESPACE_NAME
------------------------------
TESTSQL> alter tablespace test offline;
表空间已更改。
SQL> alter system checkpoint;系统已更改。
3. 运行ODU,并unload数据字典。
ODU> unload dict
get_bootstrap_dba: compat header size:12
CLUSTER C_USER# file_no: 1 block_no: 177
TABLE OBJ$ file_no: 1 block_no: 241
CLUSTER C_OBJ# file_no: 1 block_no: 49
CLUSTER C_OBJ# file_no: 1 block_no: 49
found IND$’s obj# 19
found IND$’s dataobj#:2,ts#:0,file#:1,block#:49,tab#:3
found TABPART$’s obj# 230
found TABPART$’s dataobj#:230,ts#:0,file#:1,block#:3313,tab#:0
found INDPART$’s obj# 234
found INDPART$’s dataobj#:234,ts#:0,file#:1,block#:3377,tab#:0
found TABSUBPART$’s obj# 240
found TABSUBPART$’s dataobj#:240,ts#:0,file#:1,block#:3473,tab#:0
found INDSUBPART$’s obj# 245
found INDSUBPART$’s dataobj#:245,ts#:0,file#:1,block#:3553,tab#:0
found IND$’s obj# 19
found IND$’s dataobj#:2,ts#:0,file#:1,block#:49,tab#:3
found LOB$’s obj# 156
found LOB$’s dataobj#:2,ts#:0,file#:1,block#:49,tab#:6
found LOBFRAG$’s obj# 258
found LOBFRAG$’s dataobj#:258,ts#:0,file#:1,block#:3761,tab#:0
4. 获取TEST用户下的T1表,也就是我们要恢复的表的信息:
ODU> desc test.t1
Object ID:33547
Storage(Obj#=33547 DataObj#=33549 TS#=11 File#=10 Block#=1400 Cluster=0)NO. SEG INT Column Name Null? Type
--- --- --- ------------------------------ --------- ------------------------------
1 1 1 OWNER VARCHAR2(30)
2 2 2 OBJECT_NAME VARCHAR2(128)
3 3 3 SUBOBJECT_NAME VARCHAR2(30)
4 4 4 OBJECT_ID NUMBER
5 5 5 DATA_OBJECT_ID NUMBER
6 6 6 OBJECT_TYPE VARCHAR2(18)
7 7 7 CREATED DATE
8 8 8 LAST_DDL_TIME DATE
9 9 9 TIMESTAMP VARCHAR2(19)
10 10 10 STATUS VARCHAR2(7)
11 11 11 TEMPORARY VARCHAR2(1)
12 12 12 GENERATED VARCHAR2(1)
13 13 13 SECONDARY VARCHAR2(1)
从上面的输出中,我们可以看到,TEST.T1表所在的表空间号为11,数据段头部为10号文件的1400号块。
5. 接下来用ODU扫描表空间的extent:
ODU> scan extent tablespace 11
scanning extent…
scanning extent finished.
6. 我们使用ODU来确定T1表原来的data object id。一般来说,数据段的数据块,一般是在段头后面相邻的块中。但是我们可以从段头来确认:
ODU> dump datafile 10 block 1400
Block Header:
block type=0×23 (ASSM segment header block)
block format=0×02 (oracle 8 or 9)
block rdba=0×02800578 (file#=10, block#=1400)
scn=0×0000.00286f2d, seq=4, tail=0×6f2d2304
block checksum value=0×0=0, flag=0
Data Segment Header:
Extent Control Header
-------------------------------------------------------------
Extent Header:: extents: 1 blocks: 5
last map: 0×00000000 #maps: 0 offset: 668
Highwater:: 0×02800579 (rfile#=10,block#=1401)
ext#: 0 blk#: 3 ext size:5
#blocks in seg. hdr’s freelists: 0
#blocks below: 0
mapblk: 0×00000000 offset: 0
--------------------------------------------------------
Low HighWater Mark :
Highwater:: 0×02800579 ext#: 0 blk#: 3 ext size: 5
#blocks in seg. hdr’s freelists: 0
#blocks below: 0
mapblk 0×00000000 offset: 0
Level 1 BMB for High HWM block: 0×02800576
Level 1 BMB for Low HWM block: 0×02800576
--------------------------------------------------------
Segment Type: 1 nl2: 1 blksz: 2048 fbsz: 0
L2 Array start offset: 0×00000434
First Level 3 BMB: 0×00000000
L2 Hint for inserts: 0×02800577
Last Level 1 BMB: 0×02800576
Last Level 1I BMB: 0×02800577
Last Level 1II BMB: 0×00000000
Map Header:: next 0×00000000 #extents: 1 obj#: 33549 flag: 0×220000000
Extent Map
-------------------------------------------------------------
0×02800576 length: 5Auxillary Map
-------------------------------------------------------------
Extent 0 : L1 dba: 0×02800576 Data dba: 0×02800579
-------------------------------------------------------------Second Level Bitmap block DBAs
-------------------------------------------------------------
DBA 1: 0×02800577
从上面的输出中的“Extent 0 : L1 dba: 0×02800576 Data dba: 0×02800579”可以看到,段的第1个数据块的RDBA为0×02800579,也就是10号文件的1401块。
我们dump第10号文件的1401块头,来得到表T1原来的data object id:
ODU> dump datafile 10 block 1401 header
Block Header:
block type=0×06 (table/index/cluster segment data block)
block format=0×02 (oracle 8 or 9)
block rdba=0×02800579 (file#=10, block#=1401)
scn=0×0000.00285f2b, seq=2, tail=0×5f2b0602
block checksum value=0×0=0, flag=0
Data Block Header Dump:
Object id on Block? Y
seg/obj: 0×830b=33547 csc: 0×00.285f21 itc: 3 flg: E typ: 1 (data)
brn: 0 bdba: 0×2800576 ver: 0×01Itl Xid Uba Flag Lck Scn/Fsc
0×01 0xffff.000.00000000 0×00000000.0000.00 C--- 0 scn 0×0000.00285f21
0×02 0×0000.000.00000000 0×00000000.0000.00 ---- 0 fsc 0×0000.00000000
0×03 0×0000.000.00000000 0×00000000.0000.00 ---- 0 fsc 0×0000.00000000
Data Block Dump:
================
flag=0×0 --------
ntab=1
nrow=16
frre=-1
fsbo=0×32
ffeo=0×145
avsp=0×113
tosp=0×113
可以看到,T1表原来的data object id就是33547。
7. 使用ODU来unload数据:
ODU> unload table test.t1 object 33547
Unloading table: T1,object ID: 33547
Unloading segment,storage(Obj#=33547 DataObj#=33547 TS#=11 File#=10 Block#=1400 Cluster=0)
8. 使用sqlplus将TEST表空间ONLINE:
SQL> alter tablespace test online;
表空间已更改。
9. 使用sqlldr导入我们恢复的数据:
E:\ODU\data>sqlldr test/test control=TEST_T1.ctl
SQL*Loader: Release 9.2.0.8.0 - Production on 星期日 3月 15 15:13:56 2009
Copyright (c) 1982, 2002, Oracle Corporation. All rights reserved.
达到提交点,逻辑记录计数6502
达到提交点,逻辑记录计数13004
达到提交点,逻辑记录计数19506
达到提交点,逻辑记录计数26008
达到提交点,逻辑记录计数30071
至此,恢复数据的步骤已经完成。我们来对比一下数据,看看数据是否和被Truncate前的数据完全一样:
SQL> select * from t2 minus select * from t1;
未选定行
SQL> select * from t1 minus select * from t2;
未选定行
可以看到,数据已经完全恢复。
UPDATE :
从3.0.7版本开始,对truncate table的支持更方便。可能不用人工去找到原来的data object id,而支持使用:
unload table username.tablename object auto
ODU自动探测truncate之前的data object id,对分区表同样适用。

3月 19th, 2009 at 9:19 上午
2.5.2已经发布了,刚刚才看到,等了很久了:)
5月 16th, 2009 at 4:52 下午
建议在unload table成功之后显示共unload了多少条数据。类似于dul这样的。
DUL> unload table t2 (object_id number) storage (dataobjno 25551);
. unloading table T2 16450 rows unloaded
老熊 Reply:
5月 16th, 2009 at 4:55 下午
谢谢你的建议,我会加上这个功能。
老熊 Reply:
5月 16th, 2009 at 5:49 下午
新发布的3.0.1版已经增加了此项功能。
5月 16th, 2009 at 9:52 下午
呵呵,厉害,一个小时就发布新版了。。。
5月 20th, 2009 at 8:45 下午
马上拿来,做了两个实验,真爽!
牛!
7月 20th, 2009 at 5:26 下午
请问老熊在第四步中
ODU> desc test.t1
Object ID:33547
Storage(Obj#=33547 DataObj#=33549 TS#=11 File#=10 Block#=1400 Cluster=0)
已经查到了对象ID
第七步可以直接应用么?
老熊 Reply:
7月 20th, 2009 at 5:39 下午
@guyuanli, 不能,因为那个是obj#,显示的dataobj#也是最新的,不是TRUNCATE之前的。
使用最新的3.0.7以上的版本吧。unload table xxx object auto就可以了。
9月 29th, 2009 at 2:13 下午
今天帮开发恢复truncate的一个配置表.
非常感谢!
10月 25th, 2009 at 11:30 上午
哈哈,非常感谢老熊!
1月 13th, 2010 at 4:07 下午
请问老熊
unload table username.tablename object auto
命令不成功,
unload object object_id tablespace ts# column NUMBER NUMBER NUMBER
却成功了
是何原因啊?
ODU> desc zhao.tmp_check2
Object ID:58120
Storage(Obj#=58120 DataObj#=58120 TS#=49 File#=4 Block#=773129 Cluster=0)
NO. SEG INT Column Name Null? Type
--- --- --- ------------------------------ --------- ---------------------------
---
1 1 1 BILL_REF_NO NOT NULL NUMBER(10)
2 2 2 BILL_INVOICE_ROW NUMBER
3 3 3 FEE NUMBER
ODU> unload table zhao.tmp_check2 object auto
Auto mode truncated table.
Unloading table: TMP_CHECK2,object ID: 58120
Unloading segment,storage(Obj#=58120 DataObj#=58120 TS#=49 File#=4 Block#=773129
Cluster=0)
0 rows unloaded
ODU> unload object 58120 sample
Unloading Object,object ID: 58120, Cluster: 0
output data is in file : ‘data\ODU_0000058120.txt’
Sample result:
object id: 58120
tablespace no: 49
sampled 1073 rows
column count: 3
column 1 type: NUMBER
column 2 type: NUMBER
column 3 type: NUMBER
COMMAND:
unload object 58120 tablespace 49 column NUMBER NUMBER NUMBER
ODU> unload object 58120 tablespace 49 column NUMBER NUMBER NUMBER
Unloading Object,object ID: 58120, Cluster: 0
292787 rows unloaded
老熊 Reply:
1月 13th, 2010 at 6:11 下午
@anjidx,
unload table object auto命令通常是恢复truncate表用的,这种情况下我们需要的是实际数据中的data object id与数据字典中的data object id不同。而你测试这个则是相同的。