AWR tables are not being purged according to settings in
sys.wrm$_wr_control. Because of this the tables are accumulating more
and more rows and the segments associated with these tables become very
large. Oracle decides what rows need to be purged based on the retention
policy. There is a special mechanism which is used in the case of the
large AWR tables where we store the snapshot data in partitions. One
method of purging data from these tables is by removing partitions that
only contain rows that have exceeded the retention criteria. During the
nightly purge task, we only drop the partition if all the data in the
partition has expired. If the partition contains at least one row which,
according to the retention policy shouldn't be removed, then the
partition won't be dropped and as such the table will contain old data. If partition splits do not occur (for whatever reason), then we can
end up with a situation where we have to wait for the latest entries to
expire before the partition that they sit in can be removed. This can
mean that some of the older entries can be retained significantly past
their expiry date. The result of this is that the data is not purged as
expected. A potential solution to this issue is to manually split the
partitions of the partitioned AWR objects such that there is more chance
of the split partition being purged.You will still have to wait for all
the rows in the new partitions to reach their retention time but with
split partitions there is more chance of this happening. you can
manually split the partitions using the following undocumented command: alter session set "_swrf_test_action" = 72; To perform a single split of all the AWR partitions. SELECT owner, alter session set "_swrf_test_action" = 72; NOTE: This command will split
partitions for ALL partitioned AWR objects. It also initiates a single
split; it does not need to be disabled and will need to be repeated if
multiple splits are required. SELECT owner, set serveroutput on DBMS_WORKLOAD_REPOSITORY.DROP_SNAPSHOT_RANGE( Still have questions? Consider posting a discussion in the Database Tuning Community.Applies to:
Oracle Cloud Infrastructure - Database Service - Version N/A and later
Oracle Database Backup Service - Version N/A and later
Oracle Database Cloud Exadata Service - Version N/A and later
Oracle Database Cloud Service - Version N/A and later
Oracle Database - Enterprise Edition - Version 10.2.0.1 and later
Information in this document applies to any platform.
Symptoms
Changes
Cause
Solution
segment_name,
partition_name,
segment_type,
bytes/1024/1024/1024 Size_GB
FROM dba_segments
WHERE segment_name='WRH$_ACTIVE_SESSION_HISTORY';
segment_name,
partition_name,
segment_type,
bytes/1024/1024/1024 Size_GB
FROM dba_segments
WHERE segment_name='WRH$_ACTIVE_SESSION_HISTORY';
With smaller partitions it is expected that some will be
automatically removed when the retention period of all the rows within
each partition is reached.
As an alternative, you could purge data based upon a snapshot
range. Depending on the snapshots chosen, this may remove data that has
not yet reached the retention limit so this may not be suitable for all
cases.
The following output shows the min and max snapshot_id in each partition.
declare
CURSOR cur_part IS
SELECT partition_name from dba_tab_partitions
WHERE table_name = 'WRH$_ACTIVE_SESSION_HISTORY';
query1 varchar2(200);
query2 varchar2(200);
TYPE partrec IS RECORD (snapid number, dbid number);
TYPE partlist IS TABLE OF partrec;
Outlist partlist;
begin
dbms_output.put_line('PARTITION NAME SNAP_ID DBID');
dbms_output.put_line('--------------------------- ------- ----------');
for part in cur_part loop
query1 := 'select min(snap_id), dbid from
sys.WRH$_ACTIVE_SESSION_HISTORY partition ('||part.partition_name||')
group by dbid';
execute immediate query1 bulk collect into OutList;
if OutList.count > 0 then
for i in OutList.first..OutList.last loop
dbms_output.put_line(part.partition_name||' Min '||OutList(i).snapid||' '||OutList(i).dbid);
end loop;
end if;
query2 := 'select max(snap_id), dbid from
sys.WRH$_ACTIVE_SESSION_HISTORY partition ('||part.partition_name||')
group by dbid';
execute immediate query2 bulk collect into OutList;
if OutList.count > 0 then
for i in OutList.first..OutList.last loop
dbms_output.put_line(part.partition_name||' Max '||OutList(i).snapid||' '||OutList(i).dbid);
dbms_output.put_line('---');
end loop;
end if;
end loop;
end;
/
Once you have split the partitions and identified a partition
with a range of snap ids that can be deleted, you can free up the memory
by dropping a snapshot range than matches the high and low snap_ids for
the partition:
low_snap_id IN "1",
high_snap_id IN "36255"
dbid IN NUMBER DEFAULT NULL);
Community Discussions
「喜欢这篇文章,您的关注和赞赏是给作者最好的鼓励」
关注作者
【版权声明】本文为墨天轮用户原创内容,转载时必须标注文章的来源(墨天轮),文章链接,文章作者等基本信息,否则作者和墨天轮有权追究责任。如果您发现墨天轮中有涉嫌抄袭或者侵权的内容,欢迎发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。




