增加、删除和收缩表分区!

 -- 增加表分区:

增加表分区适应于所有的分区形式,其语法是alter table tbname add partition .....
但是,需要注意对于像list,range这种存在范围值的分区,所要增加的分区值必须要大于当前分区中的最大值(如果当前存在maxvalue或default的分区,add partition会报错,这种情况只能使用split,后面会讲到),hash分区则无此限制。

SQL> create table t_partition_range (id number,name varchar2(50))
  2  partition by range(id)(
  3  partition t_range_p1 values less than (10) tablespace tbspart01,
  4  partition t_range_p2 values less than (20) tablespace tbspart02,
  5  partition t_range_p3 values less than (30) tablespace tbspart03);

表已创建。

SQL> alter table t_partition_range add partition t_range_p4 values less than(40);

表已更改。

SQL> alter table t_partition_range add partition t_range_p5 values less than(50) tablespace tbspart04;

表已更改。

Hash和list的语法与上类似,这里不再举例。
注意:
1、对于hash分区,当你执行add partition操作的时候,oracle会自动选择一个分区,并重新分配部分记录到新建的分区,这也意味着有可能带来一些IO操作。
2、执行alter table时未指定update indexes子句:
如果是range/list分区,其local索引和global索引不会受影响;
如果是hash分区,新加分区及有数据移动的分区的local索引和glocal索引会被置为unuseable,需要重新编译。
3、复合分区完全适用上述所述规则。

 

删除表分区(drop partition)
删除表分区包含两种操作,分别是:
Ø 删除分区:alter table [tbname] drop partition [ptname];
Ø 删除子分区:alter table [tbname] drop subpartition [ptname];
除hash分区和hash子分区外,其它的分区格式都可以支持这项操作。

 

--删除分区:

SQL> select table_name,partition_name from user_tab_partitions where table_name='T_PARTITION_LIST';

TABLE_NAME                     PARTITION_NAME
------------------------------ ------------------------------
T_PARTITION_LIST               T_LIST_P1
T_PARTITION_LIST               T_LIST_P2
T_PARTITION_LIST               T_LIST_P3
T_PARTITION_LIST               T_LIST_PD

SQL> alter table t_partition_list drop partition t_list_p2;

表已更改。

SQL> select table_name,partition_name from user_tab_partitions where table_name='T_PARTITION_LIST';

TABLE_NAME                     PARTITION_NAME
------------------------------ ------------------------------
T_PARTITION_LIST               T_LIST_P1
T_PARTITION_LIST               T_LIST_P3
T_PARTITION_LIST               T_LIST_PD

 

提示,drop partition时,该分区内存储的数据也将同时删除,例如:

SQL> select PARTITION_NAME,HIGH_VALUE,TABLESPACE_NAME from user_tab_partitions where TABLE_NAME = 'T_PARTITION_LIST';

PARTITION_NAME                 HIGH_VALUE           TABLESPACE_NAME
------------------------------ -------------------- ------------------------------
T_LIST_P1                      1, 2, 3, 4, 5, 6, 7, TBSPART01
                                8, 9

T_LIST_P3                      20, 21, 22, 23, 24,  TBSPART03
                               25, 26, 27, 28, 29

T_LIST_PD                      default              TBSPART04

SQL> insert into t_partition_list values (1,'test');

已创建 1 行。

SQL> insert into t_partition_list values (21,'test');

已创建 1 行。

SQL> insert into t_partition_list values (31,'test');

已创建 1 行。

SQL> commit;

提交完成。

SQL> select * from T_PARTITION_LIST;

        ID NAME
---------- --------------------------------------------------
         1 test
        21 test
        31 test

SQL> select * from T_PARTITION_LIST partition(T_LIST_P1);

        ID NAME
---------- --------------------------------------------------
         1 test

SQL> select * from T_PARTITION_LIST partition(T_LIST_P3);

        ID NAME
---------- --------------------------------------------------
        21 test

SQL> select * from T_PARTITION_LIST partition(T_LIST_PD);

        ID NAME
---------- --------------------------------------------------
        31 test

SQL> alter table t_partition_list drop partition t_list_p3;

表已更改。

SQL> select * from T_PARTITION_LIST;

        ID NAME
---------- --------------------------------------------------
         1 test
        31 test

 由于是ddl操作,这种删除也会是非常迅速的,因此如果你确认某个分区的数据都要被删除,使用drop partition会比delete更加高效。如果你的本意是希望删除掉指定的分区但保留数据,你应该使用merge partition,后面也会讲到。同样,如果你在执行该语句时没有指定update indexes子句,也会导致glocal索引的失效,至于local索引嘛,删除分区时对应的索引分区会被同时删除,但其它分区的local索引不会受到影响。

 

--收缩表分区:

Coalesce partition是个很有意思的分区功能,仅能被应用于hash分区或复合分区的hash子分区,执行之后,会自动收缩当前的表分区,比如某表当前有5个hash分区,执行alter table tbname coalesce partitions后就变成4个,再执行一次就变成3个,再执行一次就变2个,再执行一次就...........就报错了:),对于已分区的表至少要有一个分区存在的嘛!

SQL> select table_name,partition_name from user_tab_partitions where table_name='T_PARTITION_HASH';

TABLE_NAME                     PARTITION_NAME
------------------------------ ------------------------------
T_PARTITION_HASH               SYS_P24
T_PARTITION_HASH               SYS_P25
T_PARTITION_HASH               SYS_P26

SQL> alter table t_partition_hash coalesce partition;

表已更改。

SQL> select table_name,partition_name from user_tab_partitions where table_name='T_PARTITION_HASH';

TABLE_NAME                     PARTITION_NAME
------------------------------ ------------------------------
T_PARTITION_HASH               SYS_P24
T_PARTITION_HASH               SYS_P25

注意,收缩的只是分区,并不会影响到数据,但是视被收缩分区中数据的多少,收缩表分区也会涉及到IO操作。另外如果你在执行该语句时没有指定update indexes子句,收缩过程中有数据改动的分区其local索引和glocal索引都会失效,需要重新编译。

你可能感兴趣的:(增加、删除和收缩表分区!)