道家人 发表于 2024-6-9 14:53:00

Oracle分区表详解(Oracle Partitioned Tables)

当单表数据量随着时间变的越来越大时,会给数据的管理和查询带来不便。我们可以考虑对表举行分区,使用分区表特性将数据分成小块存储,可以大幅提升查询性能,管理便捷性及数据的可用性。
   目录
   一、分区表概述
   1.1 分区表概念
   1.2 何时使用分区表
   1.3 分区表的优点
   1.3.1 提升SQL查询性能
   1.3.2 提升表可管理性
   1.3.3 提升数据可用性
   二、基础分区策略
   2.1 范围分区(Range Partition)
   2.2 哈希分区(Hash Partition)
   2.3 列表分区(List partition)
   三、扩展分区策略
   3.1 复合分区(Composite Partition)
   3.2 引用分区(Reference Partition)
   3.4 系统分区(System Partition)
   四、分区表运维(Partition Maintenance)
   4.1 新增分区
   4.2 删除分区
   4.3 置换分区
   4.4 归并分区
   4.5 分裂分区
   4.6 移动分区
   4.7 重命名分区
   4.8 截断分区
   五、常用分区表视图
   5.1 dba_/all_/user_part_tables
   5.2 dba_/all_/user_tab_partitions
   5.3 dba_/all_/user_part_key_columns
   5.4 dba_/all_/user_part_col_statistics
   六、总结导图
      
   一、分区表概述

   1.1 分区表概念

   分区表就是将表在物理存储层面分成多个小的片段,这些片段即称为分区,每个分区保存表的一部分数据,表的分区对上层应用是完全透明的,从应用的角度来看,表在逻辑上依然是一个团体。
   
   每个分区都有自己的名字并可以拥有差别的存储特性,例如可以将分区保存在差别的磁盘以上分散I/O,大概分散在差别的表空间(表空间必要有类似的block size)。
   https://img-blog.csdnimg.cn/26b639fcbed4409491a9e0d9bcb233cf.png
   
   向分区表插入数据时,为了判断每条数据应该被分配至哪个分区,我们通常必要选择定义一个分区键(Partition Key)。根据每条数据分区键的值大概对其运算的结果来决定命据的分区归属,分区键可以由1或多个列构成(最多16个列).
   
   1.2 何时使用分区表

   知道了分区表的概念,那么什么情况下应该使用分区表呢?如果碰到如下几个场景你可以考虑使用分区表:
   

[*]表的巨细超过2G
[*]表中有大量的历史数据,数据存在显着的时间次序
[*]表的存储必须分散在差别的存储设备上
   
   1.3 分区表的优点

   分区表在结构和管理上比普通表更复杂,但它也有肯定的优点,紧张优点有以下3类:
   
   1.3.1 提升SQL查询性能

   对于SQL查询,当where条件涉及分区键时,可以快速定位必要扫描的分区,如许可以将数据的扫描范围限制在很小的范围,极大的提升查询性能。这个特性叫做分区裁剪(Partition Pruning)。
   另外,在多表连接(join)时,如果在每个表在连接的键上都举行了分区,那么Oracle可以将两个大表之间的连接转换成更小的分区级连接,极大提升连接速度,这个特性叫做分区连接(Partition-wise Join)。
   
   1.3.2 提升表可管理性

   使用分区表之后,原来表级别的管理操作也被分散为至“分区级”,各个分区上独立的举行运维使命,原先一个大表上的运维使命,如今可以拆开成一系列小使命分散在差别的时间窗口实行。例如,平时备份表的操作,如今可以备份单个分区。
   
   1.3.3 提升数据可用性

   当表分区后,每个分区都具有独立性。在你操作某个分区时,不会影响其他分区数据的使用,纵然某个分区因为故障不可用,也完全不会影响其他分区上运行的事件。同时分区可以存储在差别的表空间/物理介质上,分散I/O压力。
   
   二、基础分区策略

   根据差别的应用场景,你可以为表选择差别的分区策略,Oracle提供的基础分区策略有:
   

[*]范围分区(Range Partition)
[*]哈希分区(Hash Partition)
[*]列表分区(List partition)
   在基础分区策略的基础上,还有一些其他的扩展分区策略,后面再举行讨论。
   
   2.1 范围分区(Range Partition)

   范围分区根据预先定义的范围来分别分区,范围分区最适合管理类似且有显着次序的数据,根据数据的次序可以很容易规定分区范围。范围分区最典型的应用场景就是按时间对数据举行分区,以是其经常使用时间类型的分区键。
   范围分区表是通过 create table 语句的 partition by range 子句来创建的,分区的范围通过 values less than 子句指定,其指定的是分区的上限(不包含),所有大于即是指定值的数据被分配至下一个分区,除了第一个分区,每个分区的下限即前一个分区的上限:
   create table members (
id number,
name varchar2(32),
create_time date)
partition by range(create_time)
(
 partition p1 values less than (to_date('2023-02-01', 'yyyy-mm-dd')),
 partition p2 values less than (to_date('2023-03-01', 'yyyy-mm-dd')),
 partition pmax values less than (maxvalue)
);   https://img-blog.csdnimg.cn/dcb6a1a99a2f4eb889ae2e704ecc4274.png
   上面的例子中定义了3个分区:
   

[*]所有create_time小于'2023-02-01'的数据(不包含)被分配在分区p1中。
[*]所有create_time小于'2023-03-01'的数据(不包含)被分配在p2中。
[*]所有create_time大于即是'2023-03-01'的数据被分配在pmax中,如果没有这个分区,那么插入大于即是'2023-03-01'的数据时,会因为没有合适的存储分区而报错。
   
   你也可以在定义分区时指定存储特性,例如将分区分散到差别的表空间(表空间可以放到差别的物理磁盘上):
   create tablespace tbs1;
create tablespace tbs2;
create tablespace tbs3;

create table members (
id number,
name varchar2(32),
create_time date)
partition by range(create_time)
(
 partition p1 values less than (to_date('2023-02-01', 'yyyy-mm-dd')) tablespace tbs1,  -- 指定分区p1放在tbs1中
 partition p2 values less than (to_date('2023-03-01', 'yyyy-mm-dd')) tablespace tbs2,
 partition pmax values less than (maxvalue) tablespace tbs3
);   https://img-blog.csdnimg.cn/42ee37ac70354bffba81bae96cd84bd5.png
   
   2.1.1 间隔分区(Interval partition)
   间隔分区是范围分区的一个扩展,它也是通过范围来分别分区,唯一的区别是:间隔分区可以在相应分区数据插入时自动创建分区,省去了普通范围分区手动创建分区的操作。
   如果不是必要创建不规则的范围分区,那么更推荐使用间隔分区来替代范围分区,你只必要指定一个分区间隔及初始分区,后续的分区创建将由Oracle自动完成。
   间隔分区表的创建由在普通范围分区定义上新增一个interval子句创建:
   create table inv_part (
id number,
name varchar2(32),
create_time date)
partition by range(create_time)
interval (numtoyminterval(1, 'MONTH'))     -- 指定分区间隔
(
 partition p1 values less than (to_date('2023-02-01', 'yyyy-mm-dd'))
);   https://img-blog.csdnimg.cn/3d15cd7e6e4e485a927e89775c0d4cf3.png
   上面的例子指定在初始分区p1的基础上,每隔1个月创建一个分区。
   
   通过视图user_tab_partitions可以看到目前只有1个分区p1:
   select table_name, partition_name from user_tab_partitions where table_name='INV_PART';   https://img-blog.csdnimg.cn/5a5f620093444579a8477a1e5ece7b95.png
   
   我们在初始分区的上限之上插入一条数据:
   insert into inv_part values(1, 'Vincent', date '2023-02-02');
commit;
select table_name, partition_name from user_tab_partitions where table_name='INV_PART';   https://img-blog.csdnimg.cn/bc0bbcd2a5fc49daa7d3a413ed86596a.png
   在现有分区之上插入数据时,Oracle自动为我们创建了1个对应的分区SYS_P327。
   
   对于间隔分区,你也可以通过 store in 子句指定多个表空间,Oracle将以循环的方式在各个表空间中创建分区。
   create table multi_tbs (
id number,
name varchar2(32),
create_time date)
partition by range(create_time)
interval (numtoyminterval(1, 'MONTH'))store in (tbs1, tbs2, tbs3)
(
partition p1 values less than (to_date('2023-02-01', 'yyyy-mm-dd')) tablespace tbs1
);   https://img-blog.csdnimg.cn/1c095694b6484039914c907f4ba91dc3.png
   
   查询初始分区的所属表空间:
   select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='MULTI_TBS';   https://img-blog.csdnimg.cn/032ab9816633400586d7815ea356803d.png
   
   插入两条数据,触发自动创建新的分区:
   insert into multi_tbs values(1, 'Vincent', date '2023-02-02');
insert into multi_tbs values(2, 'Victor', date '2023-03-02');
commit;

select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='MULTI_TBS';   https://img-blog.csdnimg.cn/8a0d7c8b13394070ba7f95c762b3a9dc.png
   可以看到Oracle自动以循环的方式在3个表空间中创建了分区。
   
   2.2 哈希分区(Hash Partition)

   哈希分区是对指定的分区键(Partition Key)运行哈希算法来决定命据存储在哪个分区。哈希分区会随机的将数据分配到各个分区中,并尽量平均,保证各个分区的巨细差不多一致。
   由于数据是随机分布,以是哈希分区并不适合管理有显着时间次序的历史数据。它更适合必要将数据平均的分布到各个差别存储设备上的场景。同时在选用哈希分区时建议满足下列条件:
   

[*]选取分区键时尽量选取唯一列(Unique)或列中有大量唯一值(Almost Unique)的列。
[*]创建哈希分区时,分区的数量尽量是2的幂,例如2,4,8,16等。
   
   哈希分区表是通过 create table 语句的 partition by hash 子句来创建的,创建时你可以显式的指定每个分区名称,所属表空间。
   create table hash_part1 (
id number,
name varchar2(32))
partition by hash(id)
(
partition p1 tablespace tbs1,
partition p2 tablespace tbs2
);   https://img-blog.csdnimg.cn/5d8b5cb97baf4c558d57144a36341581.png
   
   也可以仅指定哈希分区的数量,此时Oracle会自动为每个分区生成名字:
   create table hash_part2 (
id number,
name varchar2(32))
partition by hash(id)
partitions 2;    -- 指定哈希分区数量,不用指定分区名   https://img-blog.csdnimg.cn/daa0fd0b502346f39724c88b41ce49e7.png
   
   你也可以用 store in 子句让分区以循环的方式建立在各个表空间中:
   create table hash_part3 (
id number,
name varchar2(32))
partition by hash(id) 
partitions 4
store in (tbs1, tbs2, tbs3);   https://img-blog.csdnimg.cn/980d45ad71a04d2da2abad740832de46.png
   
   2.3 列表分区(List partition)

   列表分区是由你为每个分区指定一系列的离散值(列表),当分区键即是特定的离散值时,数据会被放到相应的分区。列表分区可以让你自定义数据的构造方式,例如按照地域来分类数据。
   列表分区表是通过 create table 语句的 partition by list 子句来创建的,创建时你必要为每个分区指定一个列表(离散值)。
   create table list_part1 (
id number,
name varchar2(32),
city varchar2(32))
partition by list(city)
(
partition p_jiangsu values ('NanJing', 'SuZhou'),
partition p_zhejiang values('HangZhou', 'JiaXing')
);   https://img-blog.csdnimg.cn/8fc139ffbfc446a3be838c03abca3dd1.png
   
   你可以选择性的增长一个包含 default 值的分区,如许所有没有预先定义的分区键值都会放入该分区,否则会报错:
   create table list_part2 (
id number,
name varchar2(32),
city varchar2(32))
partition by list(city)
(
partition p_jiangsu values ('NanJing', 'SuZhou'),
partition p_zhejiang values('HangZhou', 'JiaXing'),
partition p_def values (default)
);   https://img-blog.csdnimg.cn/aa3ce0bd080042818a00abd29023cf4e.png
   
   列表分区建立完成后,你可以很方便的使用 alter table … modify partition … add/drop values ( … ) 来修改列表分区的枚举值:
   alter table list_part2 modify partition p_jiangsu add values('YangZhou');
alter table list_part2 modify partition p_jiangsu drop values('YangZhou');   https://img-blog.csdnimg.cn/56560856ef0842aea169e369cf7bda25.png
   如果列表分区是子分区,只必要将 modify partition 替换为 modify subpartition 即可。
   
   三、扩展分区策略

   除了前面先容的3种基础分区策略,Oracle还提供一些其他的分区策略,它们都是在基础分区策略上举行某种功能的扩充。
   
   3.1 复合分区(Composite Partition)

   复合分局,顾名思义,就是将多种分区策略联合起来使用,在基础分区的策略上,对每个分区再一次应用分区策略。例如,在基础的范围分区基础上,还可以对每个分区再次应用范围分区,即每个分区又被分别为若干个子分区。类似于中国可以分别为很多省(分区),每个省又可以分别为很多市(子分区)。
   在使用复合分区时,3种基础分区策略可以随意组合,例如,使用范围分区作为基础分区,其子分区可以使用范围、哈希、列表分区策略,即:
   

[*]范围-范围分区
[*]范围-哈希分区
[*]范围-列表分区
   其他两种分区类型同理,因此复合分区共有3*3=9种方案。
   子分区是通过原来分区策略上通过新增 subpartition子句来定义的,下面我们以范围分区(间隔分区)为基础分区,演示三种子分区的创建方式
   
   comp_part1的接纳范围-哈希分区策略:
   create table comp_part1 (
id number,
name varchar2(32),
create_time date)
partition by range(create_time) interval (numtoyminterval(1, 'MONTH'))  -- 范围分区(间隔分区)
subpartition by hash(id) subpartitions 4    -- 子分区采用哈希分区,每个范围分区再分为4个哈希分区
(
 partition p1 values less than  (to_date('2023-02-01', 'yyyy-mm-dd'))
);   https://img-blog.csdnimg.cn/736e8b711a3a49539bdb81696e7bcf33.png
   
   comp_part2的接纳范围-范围分区策略:
   create table comp_part2 (
id number,
name varchar2(32),
age number,
create_time date)
partition by range(create_time) interval (numtoyminterval(1, 'MONTH'))  -- 范围分区(间隔分区)
subpartition by range(age)     -- 子分区通过年龄进行划分
subpartition template    -- 定义子分区模板
(
 subpartition p_children    values less than (12),
 subpartition p_adolescent values less than (30),
 subpartition p_adult         values less than (60),
 subpartition p_elder         values less than (100)
)
(
 partition p1 values less than  (to_date('2023-02-01', 'yyyy-mm-dd'))
);   https://img-blog.csdnimg.cn/fb27ad7239124e278e9eee811980ca67.png
   
   comp_part3的接纳范围-列表分区策略:
   create table comp_part3 (
id number,
name varchar2(32),
sex varchar2 (32),
create_time date)
partition by range(create_time) interval (numtoyminterval(1, 'MONTH'))  -- 范围分区(间隔分区)
subpartition by list(sex)     -- 子分区通过性别进行划分
subpartition template
(
 subpartition p_man    values  ('male'),
 subpartition p_women values ('female')
)
(
 partition p1 values less than  (to_date('2023-02-01', 'yyyy-mm-dd'))
);   https://img-blog.csdnimg.cn/9f9eda905806435db75a4f0b8888668d.png
   
   3.2 引用分区(Reference Partition)

   引用分区是一种基于主-外键引用关系的分区策略,如果两张表上定义了外键引用,即两张表存在父-子关系(Parent-Child Realtionship),那么基于这种主键-外键引用关系,可以使子表继续主表的分区策略。
   引用分区特殊适合在必要自动维护子表,大概两表频繁连接查询的场景,因为他们的分区策略是类似的,两表连接通常会被转换为分区连接(partition-wise join),大大缩小连接的结果集。
   引用分区是通过partition by reference创建的。例如,下面两张表parent_table和child_table 定义了引用分区:
   create table parent_table (
id number primary key,
name varchar2(32),
create_time date)
partition by range(create_time)
interval (numtoyminterval(1, 'MONTH')) 
(
 partition p1 values less than  (to_date('2023-02-01', 'yyyy-mm-dd'))
);   https://img-blog.csdnimg.cn/8f85eca735af49c8aa840f76a95502c8.png
   
   创建子表时,如果要接纳引用分区,则定义外键的列要非空,子表会通过外键继续主表的分区方案。
   create table child_table (
id number primary key,
parent_id number not null,  -- 定义外键的列要非空
sex varchar2(32),
constraint parent_id_fk foreign key (parent_id) references parent_table(id))  -- 定义外键约束
partition by reference (parent_id_fk);   https://img-blog.csdnimg.cn/e553f55fbd004ac99aff66d10f832c0f.png
   
   下面我们验证一下引用分区的继续,通过视图 user_tab_partitions 可以看到,初始child_table也继续了主表初始分区
   select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='PARENT_TABLE';

select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='CHILD_TABLE';   https://img-blog.csdnimg.cn/bcf24a4d9310443f87293b7ffe058e12.png
   
   我们往 parent_table 中插入一条数据,触发间隔分区的自动新建分区特性:
   insert into parent_table values(1, 'Vincent', date '2023-02-02');
commit;

select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='PARENT_TABLE';   https://img-blog.csdnimg.cn/ae63da99c6114befb3a22dd2e9046dc7.png
   
   下面我们往child_table中插入一条数据:
   insert into child_table values(1, 1,'male');
commit;

select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='CHILD_TABLE';   https://img-blog.csdnimg.cn/624302b069d3425bbfe7022c71f4b284.png
   可以看到,在子表插入数据的时候,对应的分区也自动创建了出来(且分区编号都类似)。
   
   当我们在主表上删除分区时,对应的子表上的分区也被自动删除了:
   alter table parent_table drop partition SYS_P391;

select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='CHILD_TABLE';
   https://img-blog.csdnimg.cn/194344eb356d473f95c09f723e768e8d.png
   
   
   3.3 虚拟列分区(Virtual Column-based Partition)
   虚拟列分区即分区键可以定义在虚拟列上,虚拟列分区使分区键可以定义在一个表达式上,这个表达式会被保存为元数据,而列并不实际存在于数据库中。虚拟列分区可以与任何分区策略联合使用。
   下面示例中,表 virtual_part 上通过salary和bonus定义了一个虚拟列income,然后将income作为分区键:
   create table virtual_part(
id number primary key,
name varchar2(32),
salary number,
bonus number,
income as (salary + bonus))
partition by range(income)
(
 partition p1 values less than (1000),
 partition p2 values less than (5000)
);   https://img-blog.csdnimg.cn/780b84d9e5484b8dac324381ea663d46.png
   
   3.4 系统分区(System Partition)

   前先容的分区策略都是由数据库来决定命据放在哪个分区,分区对应用都是透明的。而系统分区可以仅建立一个分区表,但不指定分区策略,因此它没有分区键和分区规则。系统分区对上层应用不是透明的,应用往系统分区插入数据时,SQL必须显式的指定分区名,否则会报错。
   系统分区通过 create table 的 partition by system 子句创建,后续只必要定义分区,不必要分区键:
   create table system_part (
id number primary key,
name varchar2(32))
partition by system(
partition p1,
partition p2
);   https://img-blog.csdnimg.cn/e0edc4f782b9458fb7b794264b94c221.png
   
   系统分区的数据存储完全由应用决定,因此在插入数据时,必须显示指定命据保存的分区:
   insert into system_part values (1, 'Vincent');     https://img-blog.csdnimg.cn/10e102a8fc754477a1735f97d5dacd9d.png
   仅通过表名插入数据时报错:系统分区还必要提供分区扩展名
   insert into system_part partition(p1) values (1, 'Vincent');    https://img-blog.csdnimg.cn/ddd61ed69e564b17b1f34905c4c0ee6d.png
   插入时显式指定分区,插入乐成。
   
   四、分区表运维(Partition Maintenance)

   在一样平常运行中,我们偶然候还必要对分区表举行一些维护操作,下面是一些常见的运维案例。
   
   4.1 新增分区

   手动新增分区,差别的分区类型操作轻微有些差别。注意间隔分区和引用分区的分区都是自动创建的,因此它们无法手动新增分区。
   
   范围分区可以使用alter table … add partition 手动新增分区,注意仅可以在范围分区最大范围的上面新增分区,如果已经定义了最大值分区(maxvalue)大概想要在中间插入一个分区,则只可以使用分裂分区来完成(后面会先容):
   alter table members drop partition pmax; --由于建表时定义了p_max,要先删除才能演示,实际应用中要注意p_max分区是否有数据
   alter table members add partition p3 values less than (to_date('2023-04-01', 'yyyy-mm-dd'));   https://img-blog.csdnimg.cn/29e447a73302427d86820d50e146f33b.png
   
   哈希分区直接alter table … add partition 即可,你可以指定分区名,也可以不指定分区名,数据会重新在各分区中举行分布,大概必要一些时间:
   alter table hash_part1 add partition p3 tablespace tbs3;

alter table hash_part2 add partition tablespace tbs3;   https://img-blog.csdnimg.cn/b7639d1667e64542a0fa9b9184e82e06.png
   
   列表分区直接 alter table … add partition 新增一个分区定义:
   alter table list_part1 add partition p_anhui values('HeFei', 'ChuZhou');   https://img-blog.csdnimg.cn/7291f6b7de6c40ac96f0cc3be76b0b97.png
   
   
   4.2 删除分区

   使用 alter table … drop partition 可以删除指定的分区,对于范围分区、间隔分区,列表分区,直接指定要删除的分区名即可,间隔分区固然无法显式新增分区,但是可以显式删除:
   alter table members drop partition p3;    https://img-blog.csdnimg.cn/c3499d3f9b0d4e70bf15127c37c14393.png
   
   引用分区无法显式删除,因为它的分区策略继续自父表,只有当父表删除分区时,子表上的引用分区才会级联删除(前面演示过)。
   对于哈希分区,我们无法直接删除分区。如果要减少分区的数量,必须接纳一个叫 coalesce partition (融合分区)的操作,下面的示例会将哈希分区的数量减少1个。这个操作固然减少了一个分区,但是并不会丢失数据,数据会在剩下的分区中重新分布。
   alter table hash_part1 coalesce partition;   https://img-blog.csdnimg.cn/3e43eda60184492ab77a2f98c69dd619.png
   
   
   4.3 置换分区

   置换分区指可以用一个非分区表与分区表的某个分区/子分区举行置换(数据段交换)。使用置换分区可以快速将数据载入大概移出分区表,且置换分区操作没有类型限制,所有的分区策略都可以使用此特性。
   要置换分区,起首你要创建一个与分区表结构一样的非分区表,我们以前面的范围分区表members作为示例:
   select table_name, partition_name, tablespace_name from user_tab_partitions where table_name='MEMBERS';   https://img-blog.csdnimg.cn/f132dd947c0540858d9a1c20d4a06c21.png
   
   创建一个与members结构一样的表,并插入几条测试数据,我们筹划置换members分区p2,但是第二条数据我们插入一条违反该分区规则(create_time <'2023-03-01')的数据。
   create table mem_ext (
id number,
name varchar2(32),
create_time date);

insert into mem_ext values (3, 'exchanged_data', date '2023-02-01');
insert into mem_ext values (4, 'exchanged_data', date '2023-03-01');
commit;   https://img-blog.csdnimg.cn/0ca6b27108114138a05f21a5f6040890.png
   如果是12cR2以上的版本,你还可以用 create table … for exchange with table … 语句来快速创建一个与分区表完全匹配的非分区表:
   create table mem_ext for exchange with table members;   
   将mem_ext表与members表的p2分区举行置换:
   alter table members exchange partition p2 with table mem_ext;    -- 由于预先插入违反分区规则的数据导致报错

alter table members exchange partition p2 with table mem_ext without validation;   https://img-blog.csdnimg.cn/ad38969107b147df9d455bc324987ef9.png
   如果置换的分区中有不符合分区规则的数据(第二条),可以用 without validation 子句跳过数据验证(仅更新数据字典)。
   
   当交换分区大概更新分区键时,大概会导致数据的分区归属变革(下面第一个报错),这时候Oracle就必要在差别分区移动数据,我们可以在建表的时候开启行移动(row movement),大概手动打开,如许当分区键被更新且必要移动分区时,Oracle会自动将数据移动到正确的分区:
   update members set create_time='2023-03-03 00:00:00' where id=3;  -- 更新分区键会导致切换分区,报错

alter table members enable row movement;

update members set create_time='2023-03-03 00:00:00' where id=3;  -- 分区键更新后,数据会被移动到正确的分区   https://img-blog.csdnimg.cn/e9d11709503c403ea19d0d3b4a841f2a.png
   
   4.4 归并分区

   使用 alter table 的 merge partition/subpartion 子句,你可以将两个分区归并成一个。归并分区仅适用于范围、间隔、列表分区类型,哈希和引用分区不适用。
   对于范围分区,你只能将相邻两个的分区举行归并,且只能归并到边界高的分区,例如下面,由于p2分区上限更高,只能将分区p1归并至p2,不能将p2归并至p1:
   alter table members merge partitions p1, p2 into partition p2 update indexes;   https://img-blog.csdnimg.cn/bd1c8f46730b463dafb02bd8d951131e.png
   归并分区时,建议带上update indexes来更新索引,或归并后重修。
   
   间隔分区限制同范围分区,你也只能归并相邻的分区,而且归并还回会导致所有低于归并分区的间隔分区都转换为范围分区,归并分区的上沿就是范围分区和间隔分区的分界点,以下面的interval_part表示例,每月1个分区,我们插入数据让3、7、8,11月的间隔分区创建出来
   create table interval_part (
id number,
name varchar2(32),
create_time date)
partition by range(create_time)
interval (numtoyminterval(1, 'MONTH'))
(
 partition p1 values less than (to_date('2023-01-01', 'yyyy-mm-dd'))
);


insert into interval_part values(1,'abc', date '2023-03-10');
insert into interval_part values(1,'abc', date '2023-07-10');
insert into interval_part values(1,'abc', date '2023-08-10');
insert into interval_part values(1,'abc', date '2023-11-10');
commit;   https://img-blog.csdnimg.cn/96304788b8c24dea8969072bdc5a8ae0.png
   
   可以看到我们插入数据触发的新建分区属于间隔分区(interval=YES):
   select table_name, partition_name, interval from user_tab_partitions where table_name='INTERVAL_PART';   https://img-blog.csdnimg.cn/05645415f1444c4fb507180068138aa9.png
   
   下面将相邻的7,8月分区举行归并(SYS_448, SYS_P449):
   alter table interval_part merge partitions for (to_date('2023-07-10', 'yyyy-mm-dd')), for(to_date('2023-08-10', 'yyyy-mm-dd')) ;

select table_name, partition_name, interval from user_tab_partitions where table_name='INTERVAL_PART';   https://img-blog.csdnimg.cn/09813e13560c4be8bba927beba96e1db.png
   可以看到7,8月分区SYS_448, SYS_P449消失了,生成了一个新的分区SYS_P451,原先边界范围在归并分区之下的3月分区(SYS_P447)也被转换成了范围分区(interval=NO),而归并分区之上11月的分区(SYS_P450)依然是间隔分区(interval=YES)。
   
   列表分区由于分区之间没有次序,因此你可以归并恣意两个分区,归并后的分区包含两个分区的所有数据,以下面list_part表举例:
   create table list_part (
id number,
name varchar2(32))
partition by list(name)
(
partition p1 values ('a', 'b'),
partition p2 values('c', 'd'),
partition p3 values('e', 'f')
);   https://img-blog.csdnimg.cn/6f61e3d4f1d5486199177f4ec1d2e136.png
   
   我们将不相邻的分区p1,p3归并成了p_merged:
   select table_name, partition_name from user_tab_partitions where table_name='LIST_PART';

alter table list_part merge partitions p1,p3 into partition p_merged;

select table_name, partition_name from user_tab_partitions where table_name='LIST_PART';   https://img-blog.csdnimg.cn/73825c4ab58c438ca3a28e3a88ccae72.png
   
   4.5 分裂分区

   当某个分区过大时,你大概想要将它分裂成2个分区。分裂分区是归并分区的逆向操作,和归并分区的限制一样,分裂分区也仅适用于范围、间隔、列表分区类型,哈希和引用分区不适用。
   分裂操作会重新将数据在2个分区中举行分布,如今以上面一节归并的分区为示例,再将它们分开。
   
   分裂范围分区,我们必要指定一个分裂点(包含在分区内),整个分区将以这个分裂点为边界拆分为2个分区,分裂点会作为第一个分区的上限(不包含),下面示例将范围分区p2拆分为p1和p2:
   alter table members split partition p2 at (to_date('2023-02-01', 'yyyy-mm-dd')) into (partition p1, partition p2) update indexes;   https://img-blog.csdnimg.cn/ed6ec05ace67491d8f3a3b3fbcccfc45.png
   
   分裂间隔分区和分裂范围分区类似,我们也必要指定一个分裂点。且分裂间隔分区和和归并间隔分区一样,也会导致所有低于被分裂分区上限的间隔分区都转换为范围分区,被分裂分区的上限即范围分区和间隔分区的分界点。我们将上面示例的最后一个间隔分区 - 11月的分区(SYS_P450)从11月15号分裂为2个分区:
   select table_name, partition_name, interval from user_tab_partitions where table_name='INTERVAL_PART';

alter table interval_part split partition for(date '2023-11-10') at (date '2023-11-15') update indexes;

select table_name, partition_name, interval from user_tab_partitions where table_name='INTERVAL_PART';   https://img-blog.csdnimg.cn/d3758ab6cc9b4a3a8098e654f6298a7d.png
   分区SYS_P450分裂成了SYS_P467和SYS_P468,同时低于原分区上限的所有分区都会被转换为范围分区(interval=NO)。
   
   分裂列表分区,你必要指定必要分裂出去的值,这些指定的值会分配到第一个分区,原分区剩余的值会分配到第二个分区。
   在上面一节列表分区归并操作中,我们将p1和p3归并成了p_merged,如今再将它们分开:
   select table_name, partition_name, high_value from user_tab_partitions where table_name='LIST_PART';

alter table list_part split partition p_merged values('a', 'b') into
(
 partition p1,
 partition p3
);

select table_name, partition_name, high_value from user_tab_partitions where table_name='LIST_PART';   https://img-blog.csdnimg.cn/1399eee61c5b49e6acc223d70f72d499.png
   观察分裂前后的分区枚举值,我们指定'a', 'b'被分裂出去,那么它们将被放入p1,剩余的值会被放入p3。
   
   4.6 移动分区

   移动分区可以让你随意将某个分区移动其他表空间,这种情况通常用在必要将分区迁移到另一个存储设备上。同时也可以顺便对分区举行一些其他操作,例如压缩。所有类型的分区策略都支持移动分区。
   要移动分区至其他表空间,使用alter table的 move partition 子句,:
   alter table interval_part move partition p1 tablespace tbs1 update indexes compress;   https://img-blog.csdnimg.cn/df3e83c6135c4cc8aa123f7bac5ceb0b.png
   移动分区实际是在新目的地新建一个分区,并将原分区删除(drop),纵然目的地是类似的表空间也是云云。
   
   4.7 重命名分区

   你可以用 alter table … rename partition … to … 来给指定的分区重命名,重命名没有限制,所有分区策略都可以使用:
   alter table interval_part rename partition sys_p447 to p2;   https://img-blog.csdnimg.cn/d3e0c11efb7d40d1918ab073814c7bee.png
   
   4.8 截断分区

   必要彻底扫除某个分区数据时,你可以用 alter table … truncate partition … 来彻底扫除该分区的数据(所有分区策略都适用)。
   alter table interval_part truncate partition p2 update indexes;   https://img-blog.csdnimg.cn/7ced2c77261549dc96ba1f28e7d39f63.png
   
   五、常用分区表视图

   分区表有一组相关视图,可以供我们查询分区信息,例如前面用到的user_table_partitions,这些视图都有三个级别,分别以dba_,all_,user_开头:
   

[*]dba_ 开头的视图可以查询所有信息
[*]all_ 开头的时候可以查询有权限访问的信息(归属自己 + 被赋权的)
[*]user_ 开头的视图可以查询归属自己对象的信息
   
   5.1 dba_/all_/user_part_tables

   该组视图显示表级别的分区信息(每个分区表一条数据):
   select * from all_part_tables;   https://img-blog.csdnimg.cn/614cb0c25fc74750b77fc0ad10ac2e35.png
   紧张字段含义表明:
   https://img-blog.csdnimg.cn/7787dceeb6a643dc99877cf54ae945da.png
   
   5.2 dba_/all_/user_tab_partitions

   该组视图显示分区级别的分区信息(每个分区一条数据):
   select * from all_tab_partitions;   https://img-blog.csdnimg.cn/68e4c1e19fba4d6980b89573ff23487a.png
   紧张字段含义表明:
   https://img-blog.csdnimg.cn/14da3fa4fc9f4f89bcebe4ce3808ad88.png
   另外 dba_/all_/user_tab_subpartitions 视图显示信息类似,显示子分区级别的信息。
   
   5.3 dba_/all_/user_part_key_columns

   该组视图显示分区键信息:
   select * from all_part_key_columns;   https://img-blog.csdnimg.cn/2a8dc3bf92f745769e24f42c76e4e7d3.png
   紧张字段含义表明:
   https://img-blog.csdnimg.cn/7f8d7152bf2b47819a206531e877661c.png
   另外 dba_/all_/user_subpart_key_columns 视图显示信息类似,显示子分区级别的信息。
   
   5.4 dba_/all_/user_part_col_statistics

   改组视图显示列相关的统计信息
   select * from all_part_col_statistics;   https://img-blog.csdnimg.cn/86798466d3c6479488524ba19194bf21.png
   紧张字段含义表明:
   https://img-blog.csdnimg.cn/84dac576e7244337800e89d21fc79d67.png
   另外 dba_/all_/user_subpart_col_statistics 视图显示信息类似,显示子分区级别的信息。
   
   六、总结导图

   https://img-blog.csdnimg.cn/96c1ecef61c54032a8ea5e226497eaf5.png
   
   
   
   

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: Oracle分区表详解(Oracle Partitioned Tables)