Oracle如何从表中删除少数分区数据

Bol*_*nah 4 oracle partition

我有一个大表,其中有很多数据划分为多个分区。我想保留一些分区,但删除表中的其余数据。我尝试搜索类似的问题,但在 stackoverflow 中找不到。在 Oracle 中编写查询以实现相同目的的最佳方法是什么?

APC*_*APC 5

从特定分区中删除数据很容易:此语句清除了 2012 年 2 月的所有数据:

delete from t23 partition (feb2012);
Run Code Online (Sandbox Code Playgroud)

一个更快的方法是截断分区:

alter table t23 truncate partition feb2012;
Run Code Online (Sandbox Code Playgroud)

这里有两个潜在的障碍:

  1. 如果我们有引用表的外键,Oracle 不会让我们截断分区。
  2. 该操作使任何分区索引无效,因此我们需要在之后重建它们。

此外,它是 DDL,所以没有回滚。

如果我们再也不想存储该月的数据,我们可以删除分区:

alter table t23 drop partition feb2012;
Run Code Online (Sandbox Code Playgroud)

当我们想要切换多个分区并且我们不喜欢所有类型的输入时,问题就出现了。我们不能参数化分区名称,因为它是一个对象名称而不是一个变量(没有引号)。所以只留下动态SQL。

由于您想删除大部分数据但保留分区结构,因此截断分区是最佳选择。请记住使任何完整性约束无效(并在之后恢复它们)。

declare
    stmt varchar2(32767);
begin
    for lrec in ( select partition_name
                  from user_tab_partitions
                  where table_name = 'T23'
                  and partition_name like '%2012'
                )
    loop
        stmt := 'alter table t23 truncate partition '
                    || lrec.partition_name
                  ;
        dbms_output.put_line(stmt);
        execute immediate stmt;
    end loop;
end;
/
Run Code Online (Sandbox Code Playgroud)

您绝对应该首先运行循环并execute immediate注释掉 call,这样您就可以看到您的 WHERE 子句正在选择哪些分区。显然,您有备份并且可以恢复您不想删除的数据。但是进行恢复的最快方法是不需要恢复。

然后运行此查询以查看您应该重建哪些分区:

select ip.index_name, ip.partition_name, ip.status 
from user_indexes i
     join user_ind_partitions ip
      on  ip.index_name = i.index_name
where i.table_name = 'T23'
and ip.status = 'UNUSABLE';
Run Code Online (Sandbox Code Playgroud)

您可以以类似的方式自动执行重建语句。


“我正在考虑将我需要的分区的数据复制到临时表中并截断原始表并将数据从临时表复制回原始表。”

那是另一种做事方式。使用交换分区可能会很快。它也可能更慢。它还取决于诸如外键和索引之类的东西,以及已删除分区与保留分区的比率。如果性能很重要和/或您需要定期进行此操作,那么您应该对各种选项进行基准测试,看看哪种最适合您。