引言
人为的错误是数据库系统失败的重要原因之一根据调查约%的系统问题是操作失误或者用户错误引起的这些人为的错误又特别难以避免传统上当发生数据丢失数据错误问题时解决的主要方法就是数据的导入/导出备份/恢复技术这些方法都需要发生数据错误之前有一个正确的备份才能进行恢复恢复时不取决于错误程度而只取决于备份/恢复策略这种方法既耗时又使数据库系统不能提供服务对于一些用户偶然地删除数据这类小错误来说显得有些大材小用那么如何来恢复这种偶然的错误操作造成的数据丢失呢?从Oracle i开始提供了基于回滚段的闪回查询(Flashback Query)功能可用于恢复错误的DML操作在Oracle g中对闪回查询做了较大改进不再局限于闪回查询还可用于恢复错误的DDL(Drop)操作闪回表闪回数据库等
Oracle i的闪回查询概述
Oracle i的闪回查询功能
在Oracle i之前如果用户错误操作数据后除了不完全恢复外没有好的解决办法Oracle i中提供闪回查询由一个新的包DBMS_FLASH来实现用户使用闪回查询可以及时取得误操作DML(DeleteUpdateInsert)前某一时间点数据库的映像视图用户可以利用系统时间或系统改变号(SCNSystem Change Number)来指定这个只读视图并可以针对错误进行相应的恢复措施闪回查询功能完全依赖于自动回滚段管理(AUM)对于Drop等误操作不能恢复闪回特性可应用在以下方面
()自我维护过程中的修复当一些重要的记录被意外删除用户可以向后移动到一个时间点查看丢失的行并把它们重新插入现在的表内恢复
()恢复Email和声音Email当用户意外删除了Email或者声音信息时可以通过移回到固定时间点来恢复删除
()账号平衡状况可以查看以前的历史数据如银行外币管理中用于记录特定时间的汇率在以前汇率变更被记录在一个历史表中现在就可以通过闪回功能进行查询
()用于趋势分析的决策支持系统决策支持系统和联机分析应用必须执行一个长时间的事务使用闪回查询这些应用可以对历史数据执行分析和建模例如特定产品如矿泉水随季节变化需求情况的变化
回滚段概述
回滚段用于存放数据修改之前的位置和值回滚段的头部包含正在使用的该回滚段事务的信息回滚段的作用如下
()事务回滚当事务修改表中数据的时候该数据修改前的值(即前影像)会存放在回滚段中当用户回滚事务时Oracle将会利用回滚段中的数据前影像来将修改的数据恢复到原来的值
()事务恢复当事务正在处理的时候例程失败回滚段的信息保存在重做日志文件中Oracle将在下次打开数据库时利用回滚来恢复未提交的数据
()读一致性当一个会话正在修改数据时其它的会话将看不到该会话未提交的修改而且当一个语句正在执行时该语句将看不到从该语句开始执行后的未提交的修改(语句级读一致性)
Oracle中Delete和Commit操作的流程分析
()删除(Delete)流程
·Oracle读Block(数据块)到Buffer Cache(缓沖区)(如果该Block在Buffer中不存在)
·在Redo Log Buffer(重做日志缓沖区)中记录Delete操作的细节
·在相应回滚段段头的事物表中创建一个Undo(回滚)条目
·把将要删除的记录创建前镜像存放到Undo Block(回滚块)中
·在Buffer Cache中的相应数据块上删除记录并且标记相应的数据块为Dirty(髒)
()提交(Commit)流程
·Oracle产生一个SCN
·在回滚段事物表中标记该事物状态为Commited
·LGWR(日志读写进程) Flush Log Buffer到日志文件
·如果此时数据块仍然在Buffer Cache中那么SCN将被记录到Block Header上这被称为快速提交
·如果Dirty Block已经被写回到磁盘那么下一个访问这个Block的进程将会自回滚段中获取该事物的状态确认该事物被提交然后这个进程获得提交SCN并写回到Block Header上这被称为延迟块清除
Oracle i中闪回查询操作实例
进行闪回查询必须设置自动回滚段管理在initora设置参数UNDO_MANAGEMENT=AUTO参数UNDO_RETENTION=n决定了能往前闪回的最大时间值越大就需要越多Undo空间
例Oracle i的Flashback Query操作
()创建闪回查询用户
SQL> create user flashtest identified by flashtest;
SQL> grant connect resource to flashtest;
SQL> grant execute on dbms_flashback to flashtest;
SQL> connect flashtest/flashtest;
()创建测试表插入测试记录
SQL> create table test(id number());
SQL> insert into test values ();
SQL> insert into test values();
SQL> commit;
SQL> create table rec_date(date_scn);
注意在执行步骤或者步骤之前等待分钟
()删除记录
SQL> execute dbms_flashbackdisable;
SQL> insert into rec_date select sysdate from dual;
SQL> commit;
SQL> delete from test where id=;
SQL> commit;
通过以上的操作我们插入了两条记录并删除了其中一条记录在以下的操作中我们将通过flashback query找到删除的记录
()闪回查询
SQL> DECLARE
Restore_scn date;
BEGIN
Select date_scn into restore_scn from rec_date;
Dbms_flashbackenable_at_time (restore_scn);
END;
SQL> select * from test;
ID
可以看出虽然删除记录并提交但是通过闪回操作仍能查询到删除前的两条记录需要注意Oracle每分钟记录一次SCN并将SCN和对应时间的映射进行纪录如果原来插入的记录到做闪回操作的时间在分钟之内用基于时间的闪回查询可能得不到记录因为基于时间点的查询实际上是转化为最近的一次SCN然后从这个SCN开始进行恢复因此如果需要精确的查询可以采用基于SCN的闪回查询可精确闪回到需要恢复的时间可以通过DBMS_FLASHBACKGET_SYSTEM_CHANGE_NUMBER语句获取SCN
Oracle g的闪回查询概述
与Oracle i相比Oracle g的Flashback有了非常大的改进从普通的Flashback Query发展到了多种形式主要表现在如下几方面新特性
Flashback Database
Oracle Flashback Database特性允许通过SQL语句Flashback Database语句让数据库前滚到当前的前一个时间点或者SCN而不需要做时间点的恢复闪回数据库可以迅速将数据库回到误操作或人为错误的前一个时间点如Word中的撤消操作可以不利用备份就快速的实现基于时间点的恢复Oracle通过创建新的Flashback Logs(闪回日志)记录数据库的闪回操作如果希望能闪回数据库需要设置如下参数DB_RECOVER_FILE_DEST日志的存放位置DB_RECOVER_FILE_DEST_SIZE恢复区的大小在创建数据库的时候Oracle将自动创建恢复区但默认是关闭的需要执行alter database flashback on命令
例执行Flashback Database命令格式
SQL>flashback database to time to_timestamp(xxx);
SQL>flashback database to scn xxx
Flashback Table
Oracle Flashback Table特性允许利用Flashback Table语句确保闪回到表的前一个时间点与Oracle i中的Flashback Query相似利用回滚段信息来恢复一个或一些表到以前的一个时间点(一个快照)要注意的是Flashback Table不等于Flashback QueryFlashback Query仅仅是查询以前的一个快照点而已并不改变当前表的状态而Flashback Table将改变当前表及附属对象一起回到以前的时间点
语法
flashback table tablename to timestamp xxx或
flashback table tablename to scn xxx
注意如果需要闪回一个表需要以下条件
·需要有flashback any table的系统权限或者是该表的flashback对象权限
·需要有该表的selectinsertdeletealter权限
·必须保证该表row movement
例执行将test表闪回到年月日下午点
SQL>flashback table test to timestamp to_timestamp( ::yyyymmdd hh:mi:ss);
Flashback Drop
Oracle Flashback Drop特性提供一个类似回收站的功能用来恢复不小心被删除的表当删除表时Oracle g并不立刻释放被删除的表所占用的空间而是将这个被删除的表进行自动重命名(为了避免同类对象名称的重复)并放进回收站中所谓的回收站类似于Windows系统中的回收站是一个虚拟的容器用于存放所有被删除的对象在回收站中被删除的对象将占用创建时的同样的空间如果这个被删除的表需要进行恢复就可利用Flashback Drop功能
例进行一个删除表后恢复的简单测试
()显示回收站信息
SQL>show recyclebin;
可以看到回收站中是没有任何