zoukankan      html  css  js  c++  java
  • Oracle专家高级编程 第五章 重做和回滚

    重做

    redo log,重做日志有两种,联机和归档(数据库事务的历史记录)

    • 每个Oracle数据库至少有两个联机重做日志文件,循环使用,Oracle先往日志1写,写满专向日志2,写满转回日志文件1
    • 归档重做文件日志文件只是复制旧的,写满了的联机重做日志文件
    • 当系统写满日志文件时,ARCH进程会把联机重做日志文件复制到另一个位置
    • 当硬盘出现故障或者物理损坏时,使用归档重做日志文件实现介质恢复

    COMMIT

    commit是一个非常快的操作,不管事务大小,commit响应时间通常是"平均的"

    commit 1行花费x隔单位,commit 1000行同样花费x个单位,但是进行1000次commit一行的操作花费的是1000*x个单位

    commit提交前做好的工作

    • 在SGA中产生了回滚段(callback segment)记录
    • 在SGA中产生了修改数据块
    • 在SGA中产生了上面两条缓冲重做(undo)
    • 取决于上面3项大小和消耗的时间,一部分数据可能已经刷新到硬盘上
    • 已经获得所有的锁定

    当commit时,所有剩下的工作

    • 为事务产生一个系统改变号(System Change Number,SCN)
    • LGWR把所有余下的缓冲区中的重做日志存储到硬盘上,并在联机重做日志文件中记录SCN(当前步骤即SCSN)
    • 删除了事务条目,意味着已经提交了,在V$TRANSZCTION试图中的记录将消失
    • 释放会话占用的所有锁定,释放在入队列等待中被占用锁定的每个事项
    • 访问多个修改事务块,若它们仍在缓冲区高速缓存中,用快速模式访问和清除

    执行commit只有很少的工作,时间最长的总是LGWR执行的操作,因为它是物理硬盘的IO

    当操作进行时,LGWR连续刷新,至少:

    • 每3秒
    • 当满1/3或1MB
    • 碰到任何事物commit

    LGWR消耗的时间是由重做日志缓冲区已经释放的内容限定的

    LGWR不会存储所有工作,当操作进行时,他将增加释放后台重做日志缓冲区中的内容,可以立即释放所有重做内容

    在提交前,缓冲重做日志已经被放到硬盘上了,当commit时,必须等待

    LGWR的调用是同步的事物要等待LGWR完全写完,收到的数据存在于硬盘上

    SCN

    是Oracle保证事务顺序并从失败中恢复的简单定时机制,SCN也能保证数据库一致读取性和检查点机制,类似计数器

    回滚

    ROLLBACK 必须物理上取消已完成的工作,在ROLLBACK之前已经完成了很多工作

    • 已经在SGA产生了回滚段记录
    • 已经在SGA产生了修改数据块
    • 已经在SGA中产生了上面两条重做缓冲信息
    • 取决于上面三项大小和消耗时间,一部分上面的数据可能已经刷新到硬盘上
    • 已经获得所有锁定

    当ROLLBACK时

    撤销所有已做的修改,通过从ROLLBACK(undo)读取数据来完成,实际上是逆向操作

    • 如果插入了一行,ROLLBACK删除它
    • 如果更新了一行,ROLLBACK将逆向更新
    • 如果删除了一行,回滚将重新插入它

    释放会话占用的所有锁定,释放在队列中等待被占用锁定的每个事项

    commit只是刷新保留在重做日志缓冲区中的数据,和ROLLBACK相比,做了非常少的工作

    产生多少重做

    重做管理是数据库中串行化的关键,在LGWR结果所有事务,要求它在它们的事务中管理重做和COMMIT

    LGWR做的越多,系统越慢

    使用动态性能试图V$MYSTAT,连接V$STATNAME,然后检索名为redo size的统计值,用V$STATNAME找到它

    select * from v$statname;

    阐述如何指定事务产生的重做量,知道要修改多少数据,可以直观估计要产生多少重做

    对于AUTOTRACE追踪的语句(INSERT,UPDATE和DELETE),可以使用AUTOTRACE

    无论使用一条语句UPDATE200行,都产生相同数量的重做,对DELETE结果也一样,无论是一条语句还是200条语句

    insert情况不太一样,对于单行插入产生的重做稍微多一些,当一行插入时,必须组织不同块中的数据,一次插入多行不是这样

    产生的重做是修改数据量的函数,如果insert2000字节的行,每行产生的重做将会稍微多余2000字节

    当update时,会产生两倍数量的重做(记录日志数据和回滚),DELETE的情况和INSERT情况相似,整个行记录在回滚段

    BEFORE触发器即使不修改行中的任何值,也会增加额外的重做信息

    使用上面的技术,会发现

    • BEFORE或AFTER触发器不影响DELETE
    • 对于BEFORE或AFTER触发器,INSERT产生同样多的额外重做信息
    • UPDATE只受BEFORE触发器的影响,而AFTER触发器不增加额外重做信息
    • 行的大小影响INSERT产生的额外重做信息的数量,不影响UPDATE

    在UPDATE时,AFTER触发器对UPDATE更有效,它根本不影响重做的产生,当且仅当必须使用BEFORE触发器时才使用它

    估计重做的量

    • 估计"事务"的大小--需要修改多少数据
    • 是否增加10%~20%的额外开销,取决于要修改的行数,行越多,额外开销越少
    • 对UPDATE,此值加倍

    为了保持索引结构,必须完成索的工作量从UPDATE到UPDATE可能是不同的,考虑触发器的缺点和代表性能的隐式操作

    如何不让重做日志产生

    redo log对数据库至关重要,某些sql语句和操作支持使用nologging,可以不产生redo log

    当使用nologging操作

    • 会产生一定数量的重做,用来保护数据字典,无法避免重做的产生,虽然可以比以前明显地少
    • nologging不能防止后面的操作产生重做
    • 在ARCHIVELOG模式的数据库中执行nologging操作之后,必须尽快备份受影响的数据文件,可以避免介质故障丢失对文件的修改

    两种方式使用nologging

    • 在sql命令中何时为止嵌入关键字nologging
    • 在nologging模式中隐式执行

    在ARCHIVELOG模式数据库中,适当使用NOLOGGING,可以通过大量减少产生的重做日志数量,加速许多操作

    不能分配一个新日志

    Sun Feb 25 10:59:55 2001
    Thread 1 cannot allocate new log, sequence 326
    Checkpoint not complete

    被要求归档,而不是设置检查点没有完成

    只要数据库试图重用联机重做日志文件,但是发现不行时,就把此消息写入服务器的alert.log文件

    当DBWR没有完成检查重做日志保护的数据,或者ARCH没有把重做日志文件复制到归档为止,就会发生这种情况

    当发现会话长时间等待,最可能碰到提示:

    切换日志文件,日志缓冲区空间,或切换日志文件设置检查点或归档没有完成

    块清除

    数据锁定会影响存储在块头上的数据,缺点是当块写一次被访问时,必须“清除”它,换句话说,删除事务的信息

    这么做会产生重做并且导致块变脏

    如果主要是小到中等的事务(OLTP联机事务处理),或是在批量操作后分析表的数据仓库,一般会"清除"块

    COMMIT时处理的一个步骤,重新访问SGA中的块;如果是可访问的,清除它们,称为"提交清除"

    强制清除

    Oracle分配块清单,这些块在事务相关的提交清单中修改过,每个块清单有20个块,Oracle根据需要尽可能多地分配

    修改过的块总和超过了缓冲区高速缓存大小的10%,Oracle会停止分配新块

    SELECT产生了重做,它还"弄脏"了这些修改过的块,导致DBWR再次写入它们,这是块清除的原因

    在OLTP(联机事务处理)系统中,所有事务都是短半块,修改几块并全部清除

    create table as select

    直接路径装载数据,直接路径插入数据,都将创建''干净"的块

    日志竞争

    最大等待事件是;日志文件同步

    正在经历重做日志竞争,它们运行得不够快,发生原因有很多种

    • 一个应用程序提交频率太高,例如提交在循环中insert,

    等待LGWR把重做日志缓冲刷新到硬盘上,LGWR可以在后台完成工作,不必等待

    当COMMIT越频繁,等待越多,假定所有事务大小合适,导致日志文件等待的通常原因是

    • 重做放到了运行慢的设备上,硬盘性能差,应该去购买更块的硬盘
    • 把重做和其他文件放到了同一设备上,重做是设计用来在专用设备上大量连续写入,其他组件,试图作为LGWR同时在这台设备上读写,就会遇到一定程度竞争
    • 用缓冲方式安装设备,使用一种成熟的文件系统,双倍的缓冲降低了运行速度
    • 用慢的技术放置重做,例如RAID读取性能好,写入性能差,

    需要至少5个专用设备创建日志,为了也能镜像归档文件,最好是6个,找出最小的,最快的,一两个大的硬盘

    重做组1 -- 硬盘1和3

    重做组2 -- 硬盘2和4

    归档 -- 硬盘5和可选的硬盘6

    把成员A和B的第一组重做日志放到"组1",把成员C和D的第二组重做日志放到"组2",如果还有第3组和第4组,放到奇数组和偶数组硬盘上

    LGWR将同时写入硬盘1和硬盘3,当这组写满时,LGWR将移到硬盘2和硬盘4

    写满后,又返回重新写入硬盘1和硬盘3

    同时ARCH处理全部联机重做日志

    LGWR在网组1上写时,ARCH在读组2,同时往归档硬盘上写

    当LGWR往组2上写时,ARCH在读组1,同时往归档硬盘上写,每个LGWR和ARCH都有自己专用的设备

    它们不会竞争其他设备,也不会相互竞争

    日志文件是最利于使用RAW硬盘的一组Oracle文件

    临时表的重做和回滚

    临时表不为它们的块产生重做,临时表上的操作是不能恢复的

    当修改临时表中的块时,这种修改不会在重做日志文件中记录

    然后临时表会产生回滚,回滚是记录到日志的,临时表会产生redo log

    • INSERT产生很少甚至没有回滚/重做
    • DELETE对临时表和对正常表产生同样多的重做
    • UPDATE对临时表产生的重做只有UPDATE对正常表产生的重做的一半

    避免删除临时表,因为如果产生回滚,DELETE的逆向操作INSERT是很麻烦的

    可以使用TRUNCATE或者只是在COMMIT或会话结束后自动清空

    分析重做

    使用Logminer,在里面找到DELETE的OBJ$,查看表何时由于"事故"被删除,且希望DBA在DROP TABLE前刚刚保存 

    回滚

    ORA-01555:snapshot too old 错误问题造成的混乱十分严重

    什么产生最多/最少的撤销

    • INSERT将产生最少数量的撤销,因为Oracle所要做的记录是"删除"行ID
    • UPDATE第二位,记录修改过的字节,只要在撤销中记录行的一小部分
    • DELETE产生最多的撤销,Oracle必须把整行的"前映像"记录到撤销段,等于将整个数据写入到回滚段

    set transaction

    选择事务要使用的回滚段,主要是为比较大的操作能够确保有大的回滚段

    对于罕见的大更新,可以使用一次

    大回滚段的一个问题是,不能阻止其他事务使用

    没有办法使一个回滚段属于单个事务,如果执行许多信息的某种批量更新,使用这种特性比较好

    在原来的地方把大表分区,然后执行并行更新

    ORA-01555: Snapshot too old

    • 对在系统上执行的工作,回滚段太小
    • 程序产生交叉commit(是上面情况的一种变种)
    • 块清除

    前两点和Oracle的一致读模式直接相关,重复使用回滚段,既回滚失败的事务又提供一致读,导致了ORA-01555

    第三点在只有一个会话的数据库中,这个会话没有修改这张表,会产生错误,未修改的表为什么产生回滚呢

    • 分析相关对象,块清除是大量UPDATE或INSERT的结果
    • 增多回滚段,可以降低当执行长时间查询时回滚数据被重写的可能性
    • 降低查询的运行时间,降低对大回滚段的需要

    事实上回滚段太小了

    如果回滚段是确定大小的,那么在查询执行期间便于反转,并且查询访问的数据可能被修改可能再次遇到ORA-01555

    必须去定回滚段的大小需要配置足够的回滚,适应长时间的查询需要

    如果从查询开始的回滚都没有重写,就可以避免ORA-01555

    产生ORA-01555,是由系统最小回滚段确定的,平均容量的回滚段,最大和最小是一样的,但也有风险,最好的做法是为装载工作估计合理的使用空间

    交叉提交

    交叉提交也可能产生ORA-01555错误,可以通过commit x 行降低回滚空间,但是系统变慢,产生更多的undo和redo,必然导致ORA-01555

    最合适的配置方法,为系统配置一个合适大小的回滚段

    延迟块清除

    Oracle确定前面事务使用的回滚段(block header),确定块头是否指出前一事务的提交状态

    为了得到由于推迟块清除产生的ORA-10555,必须满足以下条件

    已经修改并commit,块依旧没有自动清除,超过的块超过了能适应SGA块缓冲区高速缓冲的10%

    论读书
    睁开眼,书在面前
    闭上眼,书在心里
  • 相关阅读:
    WinAPI 字符及字符串函数(10): lstrcpy 复制字符串
    WinAPI 字符及字符串函数(7): IsCharLower 是否是个小写字母
    WinAPI 字符及字符串函数(9): lstrcat 合并字符串
    WinAPI 字符及字符串函数(8): IsCharUpper 是否是个大写字母
    WinAPI 字符及字符串函数(6): IsCharAlphaNumeric 是否是个文字(字母或数字)
    获取汉字编码的几种方法 回复"外来天客"的问题
    上周热点回顾(4.114.17)
    博客园电子期刊2011年3月刊发布啦
    上周热点回顾(3.213.27)
    上周热点回顾(3.73.13)
  • 原文地址:https://www.cnblogs.com/YC-L/p/14663800.html
Copyright © 2011-2022 走看看