Oracle误删除表数据后的恢复具体解释

时间:2022-09-02 08:00:51

Oracle误删除表数据后的恢复具体解释

測试环境:

SYSTEM:IBM AIX 5L                         Oracle Version:10gR2

1. undo_retention參数的查询与改动

使用show parameter undo命令查看当前的数据库參数undo_retention设置。

显演示样例如以下:

SQL> show parameter undo

NAME                                 TYPE        VALUE

------------------------------------ ----------- ------------------------------

undo_management                      string      AUTO

undo_retention                       integer     900

undo_tablespace                      string      UNDOTBS2

undo_retention(保持力),900单位是秒,即15分钟。

改动默认的undo_retention參数设置:

SQL> ALTER SYSTEM SET undo_retention=10800 SCOPE=BOTH;

System altered.

SQL> show parameter undo

NAME                                 TYPE        VALUE

------------------------------------ ----------- ------------------------------

undo_management                      string      AUTO

undo_retention                       integer     10800

undo_tablespace                      string      UNDOTBS2

undo_retention 10800,单位秒,即3小时。

2. oracle误删除表数据后的的高速恢复功能方法

2.1 方法一
通过oracle提供的回闪功能
exec dbms_flashback.enable_at_time(to_date('2011-04-15 08:21:00','yyyy-mm-dd hh24:mi:ss'));

set serveroutput on
DECLARE r_temp hr.job_history%ROWTYPE;
CURSOR c_temp IS SELECT * FROM hr.job_history;
BEGIN
OPEN c_temp;
dbms_flashback.disable;
LOOP
FETCH c_temp INTO r_temp;
EXIT WHEN c_temp%NOTFOUND;
insert into hr.job_history(EMPLOYEE_ID,JOB_ID,START_DATE,END_DATE) values (r_temp.EMPLOYEE_ID,r_temp.JOB_ID,r_temp.START_DATE,r_temp.END_DATE);
commit;
END LOOP;
CLOSE c_temp;
END;
这样的办法能够将删除的数据恢复到相应的表中,首先要保证该用户有运行dbms_flashback包的权限。

2.2 方法二
insert into hr.job_history
select * from hr.job_history as of timestamp to_timestamp('2011-04-15 08:20:00', 'yyyy-mm-dd hh24:mi:ss');
这样的方法简单,easy掌握,功能和上面的一样,此处的时间为你误操作之前的时间,最好是离误操作比較近的,由于oracle保存在回滚保持段里的数据时间有一定的时间限制,这个限制由undo_retention 这个參数值决定。

查看FIRST_CHANGE#,NEXT_CHANGE#,FIRST_TIME

SQL> set pagesize 9999

SQL> col fscn for 999999999

SQL> col nscn for 999999999

SQL> select name,FIRST_CHANGE# fscn,NEXT_CHANGE# nscn,FIRST_TIME from v$archived_log;

当前的SCN为:

SQL> select dbms_flashback.get_system_change_number fscn from dual;

FSCN

----------

3435958

使用应用用户尝试闪回

SQL> connect username/password

Connected.

现有数据:

SQL> select count(*) from hs_passport;

COUNT(*)

----------

851998

创建恢复表:

SQL> create table hs_passport_recov as select * from hs_passport where 1=0;

Table created.

选择SCN向前恢复:

SQL> select count(*) from hs_passport as of scn 12929970422;

COUNT(*)

----------

861686

尝试多个SCN,获取最佳值(假设能得知详细时间,那么能够获得准确的数据闪回)

SQL> select count(*) from hs_passport as of scn &scn;

Enter value for scn: 12929941968

old   1: select count(*) from hs_passport as of scn &scn

new   1: select count(*) from hs_passport as of scn 12929941968

COUNT(*)

----------

861684

SQL> /

Enter value for scn: 12927633776

old   1: select count(*) from hs_passport as of scn &scn

new   1: select count(*) from hs_passport as of scn 12927633776

select count(*) from hs_passport as of scn 12927633776

*

ERROR at line 1:

ORA-01466: unable to read data - table definition has changed

SQL> /

Enter value for scn: 12929928784

old   1: select count(*) from hs_passport as of scn &scn

new   1: select count(*) from hs_passport as of scn 12929928784

COUNT(*)

----------

825110

SQL> /

Enter value for scn: 12928000000

old   1: select count(*) from hs_passport as of scn &scn

new   1: select count(*) from hs_passport as of scn 12928000000

select count(*) from hs_passport as of scn 12928000000

*

ERROR at line 1:

ORA-01466: unable to read data - table definition has changed

最后选择恢复到SCN为12929941968的时间点

SQL> insert into hs_passport_recov select * from hs_passport as of scn 12929941968;

861684 rows created.

SQL> commit;

Commit complete.

数据恢复简单样例

在过去,假设用户误删/更新了数据后,作为用户并没有什么直接的方法来进行恢复,他们必须求助DBA来对数据库进行恢复,到了Oracle9i,这一难堪的局面有所改善。Oracle 9i中提供了一项新的技术手段--闪回查询,用户使用闪回查询能够及时取得误操作前的数据,并能够针对错误进行对应的恢复措施,而这一切都无需DBA干预。

3. 以下我们通过一个样例来详细说明闪回查询的使用方法

演示样例
3.1 使用闪回查询前必须确定以下两个參数:
UNDO_MANAGEMENT = AUTO
undo_retention = 10800;

这个时间能够随便设,它表示在系统中保留提交了的UNDO信息的时间,10800就是保留3小时,即180分钟。

3.2 使用闪回查询
SQL> conn /as sysdba

Connected.

SQL> drop user lsf cascade;

User dropped.

SQL> create user lsf identified by lsf;

User created.

SQL> grant connect,resource to lsf;

Grant succeeded.

SQL> grant execute on dbms_flashback to lsf;

Grant succeeded.

SQL> conn lsf/lsf

Connected.

SQL> create table T(id int, name varchar2(20));

Table created.

SQL> insert into T values(1,'lsf');

1 row created.

SQL> insert into T values(2,'lsf');

1 row created.

SQL> insert into T values(3,'lsf');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

1 lsf

2 lsf

3 lsf

SQL> set time on

10:12:50 SQL> delete from T where id=1;

1 row deleted.

10:13:02 SQL> commit;

Commit complete.

10:13:10 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

2 lsf

3 lsf

10:13:18 SQL> execute DBMS_FLASHBACK.ENABLE_AT_TIME(to_date('2011-04-15 10:12:50','YYYY-MM-DD HH24:MI:SS'));

PL/SQL procedure successfully completed.

10:13:50 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

1 lsf

2 lsf

3 lsf

10:13:57 SQL> execute DBMS_FLASHBACK.DISABLE;

PL/SQL procedure successfully completed.

10:15:48 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

2 lsf

3 lsf

3.3 使用闪回查询恢复数据
10:16:59 SQL> truncate table T;

Table truncated.

10:18:15 SQL> select * from T;

no rows selected

10:18:22 SQL> insert into T values(1,'lsf');

1 row created.

10:19:42 SQL> insert into T values(2,'lsf');

1 row created.

10:19:48 SQL> insert into T values(3,'lsf');

1 row created.

10:19:55 SQL> insert into T values(4,'lsf');

1 row created.

10:20:07 SQL> insert into T values(5,'lsf');

1 row created.

10:20:15 SQL> insert into T values(6,'lsf');

1 row created.

10:20:21 SQL> commit;

Commit complete.

10:20:26 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

1 lsf

2 lsf

3 lsf

4 lsf

5 lsf

6 lsf

6 rows selected.

10:20:56 SQL> delete T;

6 rows deleted.

10:21:27 SQL> commit;

Commit complete.

10:21:40 SQL> declare

10:22:29   2  cursor flash_recover is

10:22:43   3  select * from T;

10:22:50   4  t_recode T%rowtype;

10:23:11   5  begin

10:23:14   6  DBMS_FLASHBACK.ENABLE_AT_TIME(to_date('2011-04-15 10:20:56','YYYY-MM-DD HH24:MI:SS'));

10:24:22   7  open flash_recover;

10:24:39   8  DBMS_FLASHBACK.DISABLE;

10:24:59   9  loop

10:25:05  10  FETCH flash_recover into t_recode;

10:25:24  11  EXIT WHEN flash_recover%NOTFOUND;

10:25:45  12  insert into T values(t_recode.id,t_recode.name);

10:26:35  13  end loop;

10:26:39  14  CLOSE FLASH_RECOVER;

10:26:50  15  commit;

10:26:56  16  end;

10:26:58  17  /

PL/SQL procedure successfully completed.

10:27:00 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

1 lsf

2 lsf

3 lsf

4 lsf

5 lsf

6 lsf

6 rows selected.

我们能够已经恢复了全部的6条纪录,可是因为闪回查询的局限性,有可能不能恢复全部的6条记录,原因就在以下。

4. 局限性

4.1 闪回查询是基于SCN的,尽管我们运行的是:
DBMS_FLASHBACK.ENABLE_AT_TIME(to_date('2011-04-15 10:20:56','YYYY-MM-DD HH24:MI:SS'));
但Oracle并不会精确的这个时间点,而是ROUND DOWN到近期的一次SCN,然后从这个SCN開始进行恢复。而Oracle 9i是每五分钟记录一次SCN的,并将SCN和相应时间的映射做个纪录。

因此假设使用DBMS_FLASHBACK.ENABLE_AT_TIME来进行恢复,为了避免恢复失败,我们能够先等5分钟,然后再进行恢复。
使用DBMS_FLASHBACK.ENABLE_AT_TIME进行恢复另一个缺点,那就是在Oracle 9i中SCN和相应时间的映射信息仅仅会保留5天,因此我们无法通过DBMS_FLASHBACK.ENABLE_AT_TIME来恢复5天前的数据。假设你想使用闪回查询来恢复5天前的数据,你必须自己来确定须要恢复的SCN,然后使用DBMS_FLASHBACK.ENABLE_AT_SYSTEM_CHANGE_NUMBER(SCN_NUMBER); 来定位你的恢复时间点,以下是用法:
10:27:27 SQL> VARIABLE SCN_SAVE NUMBER;

10:32:47 SQL> EXECUTE :SCN_SAVE := DBMS_FLASHBACK.GET_SYSTEM_CHANGE_NUMBER;

PL/SQL procedure successfully completed.

10:33:24 SQL> print SCN_SAVE;

SCN_SAVE

----------

3438420

10:33:41 SQL> execute DBMS_FLASHBACK.ENABLE_AT_SYSTEM_CHANGE_NUMBER(:SCN_SAVE);

PL/SQL procedure successfully completed.

10:34:31 SQL> select * from T;

ID NAME

---------- ------------------------------------------------------------

1 lsf

2 lsf

3 lsf

4 lsf

5 lsf

6 lsf

6 rows selected.

另外,在使用DBMS_FLASHBACK.ENABLE_AT_TIME前,你必须设定你的NLS_DATE_FORMAT的精确程度,Oracle默认的是精确到天,假设你不设定,像上面的样例你不会得到预期结果。

4.2 假设你使用sysdate和DBMS_FLASHBACK.GET_SYSTEM_CHANGE_NUMBER来获取时间点或者SCN值,你必须注意它们取得都是当前的时间点和SCN值。

4.3 你仅仅能在事务開始时进入闪回查询模式,假设之前有DML操作,则必须COMMIT。

4.4 闪回查询无法恢复到表结构改变之前,由于闪回查询使用的当前的数据字典。

Oracle误删除表数据后的恢复具体解释的更多相关文章

  1. ORACLE误删除表数据或误更新/插入数据如何恢复

    工作中一不小心将本不该删除/更新/插入的数据进行删除/更新/插入了,这时候一定要尽快进行恢复. 工具/原料   PL/SQL 方法/步骤     首先新建一张测试表TEST,里面输入记录.由于删除/更 ...

  2. Pl/sql 如何将oracle的表数据导出成excel文件?

    oracle将表数据导出成excel文件的方法 1)在SQL窗体上,查询需要导出的数据 --查询数据条件-- ; 结果视图 2)在查询结果的空白处,右键选择Copy to Excel 3) 查看导出e ...

  3. oracle 关于表数据delete 后如何恢复

    今天在PL/SQL中操作不小心删掉了某个表的部分数据,这可吓坏了本猿:于是悄悄的打开电脑,赶紧找度娘帮忙.经过度娘的小爬虫帮助,几分钟就把数据恢复了. 那么表数据delete掉后怎么恢复呢? 用fla ...

  4. Oracle误删除表空间的恢复

    对于误删除表空间的恢复,本文通过基于数据库的时间点恢复和基于表空间的时间点恢复分别加以讨论 一 通过基于数据库的时间点恢复被误删除的表空间 1 需要注意的事项 a 基于数据库的时间点恢复将会回退整个数 ...

  5. oracle整表数据被误删除之寻踪

    问题描述 开发同事在在14点左右发现任务表task_info数据不正确,3个小时之前的数据消失了,数据截至时间11:38:27 问题分析 查询过dba_source,只找到一个删除该表的存储过程,而且 ...

  6. oracle复制表数据,复制表结构

    1.不同用户之间的表数据复制 2.同用户表之间的数据复制 3.B.x中个别字段转移到B.y的相同字段 4.只复制表结构 加入了一个永远不可能成立的条件1=2,则此时表示的是只复制表结构,但是不复制表内 ...

  7. oracle 复制表数据,复制表结构

    1.不同用户之间的表数据复制 对于在一个数据库上的两个用户A和B,假如需要把A下表old的数据复制到B下的new,请使用权限足够的用户登入sqlplus:insert into B.new(selec ...

  8. oracle 两表数据对比---minus

        1 引言 在程序设计的过程中,往往会遇到两个记录集的比较.如华东电网PMS接口中实现传递一天中变更(新增.修改.删除)的数据.实现的方式有多种,如编程存储过程返回游标,在存储过程中对两批数据进 ...

  9. 针对mysql delete删除表数据后占用空间不变小的问题

    开发环境 Yii1版本 MySQL PHP5.6.27 前言 物流规则匹配日志表记录订单匹配规则相关日志信息,方便管理员维护和查阅不匹配的订单,四个月时间,该日志表数据就有174G,当前,这么大的数据 ...

随机推荐

  1. 一张图轻松搞懂javascript event对象的clientX,offsetX,screenX,pageX区别

    总是会被javascript的event对象的clientX,offsetX,screenX,pageX 弄得头晕,于是决定做个图来区分一下(画得我手那个酸呀....) 先总结下区别: event.c ...

  2. centos7引导项修复

    每次装了双系统,都会发现原来的windows引导项不见了,这让我这个windows重度依赖者情何以堪,所以,必须要把我挚爱的windows给找回来. 翻看了一些网上的教程,看来这并不是一个困难的问题. ...

  3. Class create, device create, device create file (转)

    来自:http://www.hovercool.com/en/Class_create,_device_create,_device_create_file 开始写Linux设备驱动程序的时候,很多时 ...

  4. 【HDOJ】5296 Annoying problem

    LCA+RMQ.挺不错的一道题目. 思路是如何通过LCA维护费用.当加入新的点u是,费用增量为dis[u]-dis[lca(u, lower_u)] - dis[lca(u, greater_u)] ...

  5. SharePoint对象模型性能考量

    转:http://blog.csdn.net/ma_jiang/article/details/6553373 部分常见用例场景都是关于显示和编辑SharePoint列表的--可惜,这也是我们看到很多 ...

  6. 转:【Java并发编程】之十二:线程间通信中notifyAll造成的早期通知问题(含代码)

    转载请注明出处:http://blog.csdn.net/ns_code/article/details/17229601 如果线程在等待时接到通知,但线程等待的条件还不满足,此时,线程接到的就是早期 ...

  7. iOS开发之duplicate symbols for architecture x86_64错误

    [主要内容:] 1. 错误提示 2. 分析错误原因 3. 解决问题办法 一.错误提示   在我们写代码过程中可能会经常遇到这样一个错误: <span style="font-size: ...

  8. Codeforces Round &num;548 &lpar;Div&period; 2&rpar; C dp or 排列组合

    https://codeforces.com/contest/1139/problem/C 题意 一颗有n个点的树,需要挑选出k个点组成序列(可重复),按照序列的顺序遍历树,假如经过黑色的边,那么这个 ...

  9. PAT甲题题解-1025&period; PAT Ranking &lpar;25&rpar;-排序

    排序,求整体的排名和局部的排名整体排序,for循环一遍同时存储整体目前的排名和所在局部的排名即可 #include <iostream> #include <cstdio> # ...

  10. ASP&period;NET Core之项目文件简介及配置文件与IOC的使用

    原文地址:https://www.cnblogs.com/knowledgesea/p/7079880.html 序言 在当前编程语言蓬勃发展与竞争的时期,对于我们.net从业者来说,.Net Cor ...