delete archivelog all 无法彻底删除归档日志?
最近在因归档日志暴增,使用delete archivelog all貌似无法清除所有的归档日志,到底是什么原因呢?
[python]
1、演示环境
sql> select * from v$version where rownum<2;
banner
----------------------------------------------------------------
oracle database 10g release 10.2.0.3.0 - 64bit production
sql> select inst_id,instance_name from gv$instance; -->两节点rac
inst_id instance_name
---------- ----------------
1 gobo4a
2 gobo4b
sql> show parameter db_recovery -->+rev,使用了asm 存储方式
name type value
------------------------------------ ----------- -------------
db_recovery_file_dest string +rev
db_recovery_file_dest_size big integer 1g
sql> select flashback_on from v$database; -->未开启闪回特性,也就是说尽管指定了闪回区,未启用闪回特性
-->相应的,归档日志充满整个闪回区时,闪回区空间并不会被重用
flashback_on
------------------
no
2、查看及清除现有的归档日志文件
oracle@bo2dbp:~> export oracle_sid=+asm1
oracle@bo2dbp:~> asmcmd
asmcmd> cd +rev/gobo4/archivelog
asmcmd> ls
2012_10_08/
....
arch_795194241_1_10.arc
arch_795194241_1_100.arc
....
oracle@bo2dbp:~> export oracle_sid=gobo4a
oracle@bo2dbp:~> rman target /
recovery manager: release 10.2.0.3.0 - production on thu nov 29 16:23:15 2012
copyright (c) 1982, 2005, oracle. all rights reserved.
connected to target database: gobo4 (dbid=921286879)
#下面通过使用rman backup archivelog方式来删除所有的归档日志文件
rman> backup format '/install_source/rman_bak/arch_%d_%u'
2> archivelog all delete input;
starting backup at 29-nov-12
current log archived
using target database control file instead of recovery catalog
allocated channel: ora_disk_1
channel ora_disk_1: sid=1058 instance=gobo4a devtype=disk
channel ora_disk_1: starting archive log backupset
channel ora_disk_1: specifying archive log(s) in backup set
input archive log thread=1 sequence=139 recid=214 stamp=797450261
input archive log thread=1 sequence=140 recid=215 stamp=797450292
input archive log thread=1 sequence=141 recid=216 stamp=797450308
input archive log thread=1 sequence=142 recid=218 stamp=797450347
input archive log thread=1 sequence=143 recid=219 stamp=797450372
input archive log thread=1 sequence=144 recid=220 stamp=797450409
channel ora_disk_1: starting piece 1 at 29-nov-12
channel ora_disk_1: finished piece 1 at 29-nov-12
piece handle=/install_source/rman_bak/arch_gobo4_1dnrhkn4_1_1 tag=tag20121129t162806 comment=none
channel ora_disk_1: backup set complete, elapsed time: 00:02:15
channel ora_disk_1: deleting archive log(s)
archive log filename=+rev/gobo4/archivelog/arch_795194241_1_139.arc recid=214 stamp=797450261
archive log filename=+rev/gobo4/archivelog/arch_795194241_1_140.arc recid=215 stamp=797450292
archive log filename=+rev/gobo4/archivelog/arch_795194241_1_141.arc recid=216 stamp=797450308
........
piece handle=/install_source/rman_bak/arch_gobo4_1hnrhli2_1_1 tag=tag20121129t162806 comment=none
channel ora_disk_1: backup set complete, elapsed time: 00:00:09
channel ora_disk_1: deleting archive log(s)
archive log filename=+rev/gobo4/archivelog/arch_795194241_2_141.arc recid=427 stamp=800547491
archive log filename=+rev/gobo4/archivelog/arch_795194241_2_142.arc recid=429 stamp=800549193
archive log filename=+rev/gobo4/archivelog/arch_795194241_2_143.arc recid=433 stamp=800578944
archive log filename=+rev/gobo4/archivelog/arch_795194241_2_144.arc recid=437 stamp=800641679
finished backup at 29-nov-12
#再次查看依然有很多归档日志文件存在,而且都是10月23日之前的
asmcmd> pwd
+rev/gobo4/archivelog
asmcmd> ls
2012_09_30/
2012_10_09/
2012_10_10/
2012_10_11/
2012_10_12/
2012_10_13/
2012_10_14/
2012_10_15/
2012_10_16/
2012_10_17/
2012_10_18/
2012_10_22/
2012_10_23/
arch_795194241_1_100.arc
arch_795194241_1_101.arc
arch_795194241_1_102.arc
............
#再次删除日志文件,来个更狠的命令,直接delete所有的archivelog,最近新增的一个archivelog被删除
rman> delete noprompt archivelog all;
released channel: ora_disk_1
allocated channel: ora_disk_1
channel ora_disk_1: sid=1081 instance=gobo4a devtype=disk
list of archived log copies
key thrd seq s low time name
------- ---- ------- - --------- ----
453 1 294 a 29-nov-12 +rev/gobo4/archivelog/arch_795194241_1_294.arc
deleted archive log
archive log filename=+rev/gobo4/archivelog/arch_795194241_1_294.arc recid=453 stamp=800662185
deleted 1 objects
# 上面输出的结果只有一个归档日志被删除,何以故?
# 这个我们的分析一下delete noprompt archivelog all以及备份归档日志时使用的 delete input
# 回顾一下oracle控制文件以及oracle rman的的备份恢复的原理。
# 我们知道,oracle 控制文件里边记录了数据库的名字,id,创建的时间戳....一大堆的信息,当然也有不可少的归档信息以及备份信息。
# 如果不知道控制文件有什么? 那就参考:oracle 控制文件,文章尾部有给出链接。
# 其次,oracle rman的备份恢复的所有信息都依赖于两个东东,要么是控制文件,要么是恢复目录(catalog)。
# 因为所有的备份与恢复信息都会依据备份是的方式存储到这两个位置。
# 理所当然的是,对这两个东东里的备份集,镜像副本,归档日志,等等所有能备份的对象的任意操作,首先会参考这些对象的记录的信息。
# 其次是当被记录的对象发生变化时做相应的更新。
3、深度分析无法清除的原因
#先来看看gv$archived_log,如果是单实例使用v$archived_log
#从下面的查询可知,又有两个新的归档日志产生,一个从第一个instance产生,一个从第二个instance产生。
sql> select name,status,count(*) from gv$archived_log group by name,status;
name s count(*)
-------------------------------------------------- - ----------
d 444
+rev/gobo4/archivelog/arch_795194241_1_295.arc a 2
+rev/gobo4/archivelog/arch_795194241_2_150.arc a 2
# 从上面的查询可知,当前的两个节点其归档日志只有2个,其余的444个其名字都是null值。
# 看看关于视图v$archived_log中name列的解释
# archived log file name. if set to null, either the log file was cleared before it was archived or an rman backup command
# with the "delete input" option was executed to back up archivelog all (rman> backup archivelog all delete input;).
# 上面的这段话表明当前的这些日志文件要么被手动清除,要么被rman的delete input选项清除。
# 其次status列的d字段也表明了这些个名字为空的归档日志已经被deleted.也就是说有444个归档日志已经被删除了。
# 再次尝试删除归档日志,尾数为295和150的归档日志也被删除
rman> delete noprompt archivelog all;
released channel: ora_disk_1
allocated channel: ora_disk_1
channel ora_disk_1: sid=1081 instance=gobo4a devtype=disk
list of archived log copies
key thrd seq s low time name
------- ---- ------- - --------- ----
454 1 295 a 29-nov-12 +rev/gobo4/archivelog/arch_795194241_1_295.arc
455 2 150 a 29-nov-12 +rev/gobo4/archivelog/arch_795194241_2_150.arc
deleted archive log
archive log filename=+rev/gobo4/archivelog/arch_795194241_1_295.arc recid=454 stamp=800712037
deleted archive log
archive log filename=+rev/gobo4/archivelog/arch_795194241_2_150.arc recid=455 stamp=800712038
deleted 2 objects
# 查询gv$archived_log视图,表明所有现有的archivelog都已经被删除
sql> select name,status,count(*) from gv$archived_log group by name,status;
name s count(*)
-------------------------------------------------- - ----------
d 448
# 在asmcmd命令下也无法找到我们刚刚删除的归档日志文件
asmcmd> pwd
+rev/gobo4/archivelog
asmcmd> ls -l arch_795194241_1_295.arc
asmcmd: entry 'arch_795194241_1_295.arc' does not exist in directory '+rev/gobo4/archivelog/'
asmcmd> ls -l arch_795194241_2_150.arc
asmcmd: entry 'arch_795194241_2_150.arc' does not exist in directory '+rev/gobo4/archivelog/'
# 在a节点上再次切换一次
sql> alter system switch logfile;
system altered.
sql> select inst_id,name,count(*) from gv$archived_log group by inst_id,name;
inst_id name count(*)
---------- -------------------------------------------------- ----------
2 223
1 +rev/gobo4/archivelog/arch_795194241_1_296.arc 1
2 +rev/gobo4/archivelog/arch_795194241_1_296.arc 1
1 223
--上面的查询可以看到当前的一个归档日志arch_795194241_1_296.arc基于inst_id为1的有1个,而基于inst_id为2的也有一个
--而直接查询v$archived_log时只有1个当前的归档日志,实际上arch_795194241_1_296.arc文件是由第一个instance产生的。
--数字296之前的1即可以表明为第一个instance产生的。
sql> select name from v$archived_log where name='+rev/gobo4/archivelog/arch_795194241_1_296.arc';
name
--------------------------------------------------
+rev/gobo4/archivelog/arch_795194241_1_296.arc
# 关于这个地方个人认为这个应该是用于做恢复时用的。
# rac数据库在恢复时,无论多个少节点,只有所有的归档日志的集合才能完成地表述数据库的变迁。
# 此时,无论从哪个节点上看,或者说做无论从哪个节点恢复,都可以看到该归档日志。
# 而具体是哪个instance产生则由'%t'重做线程编号来判断。
#下面再来看看控制文件
sql> select * from gv$controlfile_record_section where type='archived log';
inst_id type record_size records_total records_used first_index last_index last_recid
---------- ---------------------------- ----------- ------------- ------------ ----------- ---------- ----------
1 archived log 584 224 224 149 148 456
2 archived log 584 224 224 149 148 456
# records_total:number of records allocated for the section
# 列records_total表明为当前type分配的可存储的总数,在两个instance上都为224条
# 从最近一次切换日志的查询结果可知,被删除的有223条,新增的一条为arch_795194241_1_296.arc,总条数为224条。
# 如果下次日志切换再增加一条往哪里放呢?那些已经超出缺省保留期的归档日志被覆盖,即被重用。
# 用户在控制文件中保存archived log部分的保留时间由谁来决定呢,参数control_file_record_keep_time,缺省为7天
# 这意味着7天前的归档日志和备份信息可能在控制文件中已经不存在了
sql> show parameter control_file_record_keep_time
name type value
------------------------------------ ----------- ------------------------------
control_file_record_keep_time integer 7
sql> select count (*) from v$archived_log;
count(*)
----------
224
# author : robinson
# blog : https://blog.csdn.net/robinson_0612
sql> alter session set nls_date_format='yyyymmdd hh24:mi:ss';
session altered.
# 下面的查询正好表明为什么2012_10_23和之前的日志为什么没有被删除
# 因为20121023 18:04:53之后的归档日志已经被覆盖了,所以使用delete archivelog all时是根本无法清除之前的日志的,无能为力阿。
# 对于rman下的delete archivelog all方式不会删除控制文件中对应的归档日志信息,但在控制文件中设置delete状态,
# 即v$archived_log视图的status列为deleted
sql> select min (first_time), min (completion_time), max (first_time), max (completion_time) from
2 v$archived_log;
min(first_time) min(completion_ti max(first_time) max(completion_ti
----------------- ----------------- ----------------- -----------------
20121023 18:03:12 20121023 18:04:53 20121130 12:00:26 20121130 12:14:51
sql> select min (first_time), min (completion_time), max (first_time), max (completion_time) from
2 gv$archived_log;
min(first_time) min(completion_ti max(first_time) max(completion_ti
----------------- ----------------- ----------------- -----------------
20121023 18:03:12 20121023 18:04:53 20121130 12:00:26 20121130 12:14:51
# 既然这般,如何是好啊?
# 那就直接在asmcmd命令行下删除吧。一顿狂删 rm -rf 2012_09_30/
# 莫急,莫急,一不小心删完了,我晕,ora-00254/ora-15173 archive_log directory on asm being deleted 在等候阿。
小结
a、delete archivelog all将会毫无保留的删除所有的归档日志(在控制文件中有相应记录的)
b、归档日志的信息被记录在控制文件之中,其生存期和可保留的总数也受到控制文件创建初以及参数control_file_record_keep_time限制
c、对于那些已经在控制文件中被覆盖的归档日志,该方式不起作用,使用backup archivelog all delete input同样不起作用
d、注意backup archivelog all时delete input与delete all input有些差异,前者删除仅仅被备份过的归档日志,而后者则对于多个归档位置
下的所有归档日志全部删除。
e、视图v$archived_log或gv$archived_log提供了归档日志的相关详细信息
f、建议备份归档日志后再删除。注,rac+asm下切不可使得archivedlog文件夹为空,否则,整个文件夹连同上级空目录会被删除
本文涉及到的一些参考文章:
oracle 控制文件(controlfile)
oracle 归档日志
参数control_file_record_keep_time和maxloghisotry
使用 asmcmd 工具管理asm目录及文件
ora-00254/ora-15173 archive_log directory on asm being deleted