您的位置:首页 > 博客中心 > 数据库 >

oracle 分区和分区索引

时间:2022-03-10 18:06

一、个人理解:建表时一般都会指定在一个表空间上,但是可能随着表空间扩大,查询越来越慢,分区表就是将一个表实际存在不同的表空间,oracle存储分为块,断,表空间。新建一个表,会给表分配指定大小的段,段里包含存储块,高水位线就是指段空间被撑大了。

表空间和分区的区别:

(1)表空间是逻辑存储单位,所有的表段放在表空间里。至于表的分区,也可看作一个逻辑段,属于比表空间小一级的逻辑存储单位。

(2)他们根本不是一个概念,表空间是由多个数据文件组成的,可以这么说 
块组成了段,段组成了表空间,表空间组成了数据库

快包括4k,8k,16k,32k
段包括表,表分区上每个分区等等
每个分区又可以存放在多个表空间上,可以放在不同的磁盘,增加吞吐量。

也有人补充:塊組成域,塊也可以是5K,6K……

当表中的数据量不断增大,查询数据的速度就会变慢,应用程序的性能就会下降,这时就应该考虑对表进行分区。表进行分区后,逻辑上表仍然是一张完整的表,只是将表中的数据在物理上存放到多个表空间(物理文件上),这样查询数据时,不至于每次都扫描整张表。 

二、表分区和分区索引

查询分区:Select *From user_extents WHERE partition_name=‘分区名‘;

1)创建表空间

create tablespace HRPM0

datafile ‘/oradata/misdb/HRPM0.DBF‘ size 5m autoextend on next 10m maxsize unlimited

2)删除表空间(同时把数据文件也删除)

DROP TABLESPACE data01 INCLUDING CONTENTS AND DATAFILES;

   如果不想删除数据文件:

           Drop tablespace tablespace_name;

3) 修改表空间大小

alter database datafile ‘/path/NADDate05.dbf‘ resize 100M

4)添加数据文件(在建立表空间时,若是约束了表空间的大小,那么一段时间后,这个表空间就会被装满,无法再添加其他对象。则需要给表空间添加数据文件):

Alter tablespace tablespace_name add datafile’ ‘/path/NADDate06.dbf’ size 100M;

4) 备注:

4.1).--.禁止undo tablespace自动增长

alter database datafile ‘full_path\undotbs01.dbf‘ autoextend off;

4.2).-- 创建一个新的小空间的undo tablespace

create undo tablespace undotBS2 datafile ‘full_path\UNDOTBS02.DBF‘ size 100m;

4.3).-- 设置新的表空间为系统undo_tablespace

alter system set undo_tablespace=undotBS2;

4.4).-- Drop 旧的表空间

drop tablespace undotbs1 including contents;

4.5).--查看所有表空间的情况

select * from dba_tablespaces

5)查到一个最好用的表:dict

5.1)select *from dict where table_name like ‘%PART%‘

5.2)ALL_TAB_PARTITIONS:可以查出表所对应的分区内容;

5.3)dab_tab_partitons :与上2);

5.4)dba_ind_partitons:查询分区的索引;

5.5)子分区也是一样的(dba_tab_subpartitons,dba_ind_partitons)

一、使用分区的优点:

    1、增强可用性:如果表的某个分区出现故障,表在其他分区的数据仍然可用;

    2、维护方便:如果表的某个分区出现故障,需要修复数据,只修复该分区即可;

    3、均衡I/O:可以把不同的分区映射到磁盘以平衡I/O,改善整个系统性能;

select * from user_tablespaces

 

                   <表空间->三个>

insert into PROBLEM_TICKETS values(2,‘son‘,2,sysdate+30,‘INACTIVE‘);

insert into PROBLEM_TICKETS values(3,‘son‘,3,to_date(‘2006-05-30‘,‘yyyy-mm-dd‘),‘INACTIVE‘);

insert into PROBLEM_TICKETS values(4,‘BOOKS‘,4,to_date(‘2007-06-23‘,‘yyyy-mm-dd‘),‘INACTIVE‘);

insert into PROBLEM_TICKETS values(5,‘old‘,5,to_date(‘2011-02-26‘,‘yyyy-mm-dd‘),‘ACTIVE‘);

insert intoPROBLEM_TICKETSvalues(6,‘test‘,6,to_date(‘2011-04-30‘,‘yyyy-mm-dd‘),‘INACTIVE‘);

select * from PROBLEM_TICKETS

 

                                     <查询全表>

1.4.2)

   select * from PROBLEM_TICKETS partition(PROB_ACTIVE)

 

1.4.2)

   select * from PROBLEM_TICKETS partition(PROB_INACTIVE)

 

 在测试中遇到这样的情况。如果表创建了分区,如果要删除数据文件(表空间文件),则要先删除分区,然后才能删除数据文件(但是在删除数据文件时,必须要保留一个分区才能最终删除数据文件>表空间文件,)

当然,也可以直接就删除表也行,刚所有的全删除,但是表空间文件还在!

 

1.5) 复合分区

      partition part_07 values less than(to_date(‘2006-01-01‘,‘yyyy-mm-dd‘)),

      partition part_08 values less than(to_date(‘2010-01-01‘,‘yyyy-mm-dd‘)),

      partition part_09 values less than(maxvalue)

 );

 ---测试如下:

   select *From user_tab_partitions where table_name=upper(‘dinya_test‘)

            selec *From user_tab_subpartitions where table_name=upper(‘dinya_test‘)

  

                            《图1》

 

   插入如下数据:

 insert into dinya_test values(1,12,‘BOOKS‘,sysdate);

insert into dinya_test values(2,12, ‘BOOKS‘,sysdate+30);

insert into dinya_test values(3,12, ‘BOOKS‘,to_date(‘2006-05-30‘,‘yyyy-mm-dd‘));

insert into dinya_test values(7,12, ‘BOOKS‘,to_date(‘2005-05-30‘,‘yyyy-mm-dd‘));

insert into dinya_test values(4,12, ‘BOOKS‘,to_date(‘2007-06-23‘,‘yyyy-mm-dd‘));

insert into dinya_test values(5,12, ‘BOOKS‘,to_date(‘2011-02-26‘,‘yyyy-mm-dd‘));

insert into dinya_test values(6,12, ‘BOOKS‘,to_date(‘2011-04-30‘,‘yyyy-mm-dd‘));

 select *From dinya_test:如下图

 

select *From dinya_test partition(part_07)如下图:

 

select *From dinya_test partition(part_09)

 

参照下图,按所显的子分区名,看能否查出数据:

select*Fromuser_tab_subpartitions where table_name=upper(‘dinya_test‘)

 

select *From dinya_test subpartition(SYS_SUBP62):如下图:

 

其它的查询一样。-----测试成功;

  备注:该例中,先是根据交易日期进行范围分区,然后根据交易的ID将记录散列地存储在三个表空间中。

 

1.6)复合范围列表分区:这种分区是基于范围分区和列表分区,表首先按某列进行范围分区,然后再按某列进行列表分区,分区之中的分区被称为子分区。

示例1:

Create table sales

(

Product_id varchar2(5),

Sales_date date,

Sales_cost number(10),

Status varchar2(20)

)

Partition by range(Sales_cost)

Subpartition by list(status)

(

Partition p1 values less than (1) tablespace dinya_space01

(

Subpartition p1sub1 values(‘ACTIVE‘) tablespace   dinya_space03,

Subpartition p1sub2 values(‘INACTIVE‘) tablespace dinya_space04

),

Partition p2 values less than (3) tablespace dinya_space02

 (

Subpartition p1sub3 values(‘ACTIVE‘) tablespace    dinya_space05,

Subpartition p1sub4 values(‘INACTIVE‘) tablespace dinya_space06

)

)测试如下:

insert into sales values(1,sysdate,0.1,‘ACTIVE‘);

insert into sales values(2,sysdate+30,1,‘INACTIVE‘);

insert into sales

values(3,to_date(‘2006-05-30‘,‘yyyy-mm-dd‘),2,‘INACTIVE‘);

select *From sales:

 

        Select *from sales partition(p2)

 

       SELECT * FROM SALES SUBPARTITION(p1sub4)

 

SELECT * FROM SALES SUBPARTITION(p1sub3)

                           没有数据!!

select *From dba_tab_subpartitions where table_name=‘SALES‘

 

 

 有关表分区的一些维护性操作:

一、添加分区

以下代码给SALES表添加了一个P3分区

ALTER TABLE SALES ADD PARTITION P3 VALUES LESS THAN(TO_DATE(‘2003-06-01‘,‘YYYY-MM-DD‘));

注意:以上添加的分区界限应该高于最后一个分区界限。

以下代码给SALES表的P3分区添加了一个P3SUB1子分区

ALTER TABLE SALES MODIFY PARTITION P3 ADD SUBPARTITION P3SUB1 VALUES(‘COMPLETE‘);

 

二、删除分区

以下代码删除了P3表分区:

ALTER TABLE SALES DROP PARTITION P3;

 在测试中遇到这样的情况。如果表创建了分区,如果要删除数据文件(表空间文件),则要先删除分区,然后才能删除数据文件(但是在删除数据文件时,必须要保留一个分区才能最终删除数据文件>表空间文件,)

当然,也可以直接就删除表也行,刚所有的全删除,但是表空间文件还在!

 

在以下代码删除了P4SUB1子分区:

ALTER TABLE SALES DROP SUBPARTITION P4SUB1;

注意:如果删除的分区是表中唯一的分区,那么此分区将不能被删除,要想删除此分区,必须删除表。

 

三、截断分区

截断某个分区是指删除某个分区中的数据,并不会删除分区,也不会删除其它分区中的数据。当表中即使只有一个分区时,也可以截断该分区。通过以下代码截断分区:

ALTER TABLE SALES TRUNCATE PARTITION P2;

通过以下代码截断子分区:

ALTER TABLE SALES TRUNCATE SUBPARTITION P2SUB2;

 

四、合并分区

合并分区是将相邻的分区合并成一个分区,结果分区将采用较高分区的界限,值得注意的是,不能将分区合并到界限较低的分区。以下代码实现了P1 P2分区的合并:

ALTER TABLE SALES MERGE PARTITIONS P1,P2 INTO PARTITION P2;

 

五、拆分分区

拆分分区将一个分区拆分两个新分区,拆分后原来分区不再存在。注意不能对HASH类型的分区进行拆分。

ALTER TABLE SALES SBLIT PARTITION P2 AT(TO_DATE(‘2003-02-01‘,‘YYYY-MM-DD‘))

INTO (PARTITION P21,PARTITION P22);

 

六、接合分区(coalesca)

结合分区是将散列分区中的数据接合到其它分区中,当散列分区中的数据比较大时,可以增加散列分区,然后进行接合,值得注意的是,接合分区只能用于散列分区中。通过以下代码进行接合分区:

ALTER TABLE SALES COALESCA PARTITION;

 

七、重命名表分区

以下代码将P21更改为P2

ALTER TABLE SALES RENAME PARTITION P21 TO P2;

 

九、跨分区查询

select sum( *) from (

(select count(*) cn from t_table_SS PARTITION (P200709_1)

union all

select count(*) cn from t_table_SS PARTITION (P200709_2));

 

十、查询表上有多少分区

SELECT * FROM useR_TAB_PARTITIONS WHERE TABLE_NAME=‘tableName‘

 

 

--显示数据库所有分区表的信息:

select * from DBA_PART_TABLES where table_name=upper(‘dinya_test‘)

 

 

--显示当前用户可访问的所有分区表信息:

select * from ALL_PART_TABLES

同上图

--显示当前用户所有分区表的信息:

select * from USER_PART_TABLES

同上图

--显示表分区信息 显示数据库所有分区表的详细分区信息:

select * from DBA_TAB_PARTITIONS

 

--显示当前用户可访问的所有分区表的详细分区信息:

select * from ALL_TAB_PARTITIONS

 

--显示当前用户所有分区表的详细分区信息:

select * from USER_TAB_PARTITIONS

 

--显示子分区信息 显示数据库所有组合分区表的子分区信息:

select * from DBA_TAB_SUBPARTITIONS

 

--显示当前用户可访问的所有组合分区表的子分区信息:

select * from ALL_TAB_SUBPARTITIONS

 

--显示当前用户所有组合分区表的子分区信息:

select * from USER_TAB_SUBPARTITIONS

 

--显示分区列 显示数据库所有分区表的分区列信息:

select * from DBA_PART_KEY_COLUMNS

 

--显示当前用户可访问的所有分区表的分区列信息:

select * from ALL_PART_KEY_COLUMNS

 

--显示当前用户所有分区表的分区列信息:

select * from USER_PART_KEY_COLUMNS

 

--显示子分区列 显示数据库所有分区表的子分区列信息:

select * from DBA_SUBPART_KEY_COLUMNS

 

--显示当前用户可访问的所有分区表的子分区列信息:

select * from ALL_SUBPART_KEY_COLUMNS

 

--显示当前用户所有分区表的子分区列信息:

select * from USER_SUBPART_KEY_COLUMNS

 

--怎样查询出oracle数据库中所有的的分区表

select * from user_tables a where a.partitioned=‘YES‘

 

--删除一个表的数据是

truncate table table_name;

 

--删除分区表一个分区的数据是

alter table table_name truncate partition p5;

 

注:分区根据具体情况选择。

 

表分区有以下优点:

1、数据查询:数据被存储到多个文件上,减少了I/O负载,查询速度提高。

2、数据修剪:保存历史数据非常的理想。

3、备份:将大表的数据分成多个文件,方便备份和恢复。

4、并行性:可以同时向表中进行DML操作,并行性性能提高。

================================================

 

索引:

1、一般索引:

create index index_name on table(col_name);

2、Oracle 分区索引详解

语法:Table Index

CREATE [UNIQUE|BITMAP] INDEX [schema.]index_name

ON [schema.]table_name [tbl_alias]

(col [ASC | DESC]) index_clause index_attribs

 

index_clauses:

分以下两种情况

 

1. Local Index

就是索引信息的存放位置依赖于父表的Partition信息,换句话说创建这样的索引必须保证父表是Partition

1.1 索引信息存放在父表的分区所在的表空间。但是仅可以创建在父表为HashTable或者composite分区表的。

LOCAL STORE IN (tablespace)

1.2 仅可以创建在父表为HashTable或者composite分区表的。并且指定的分区数目要与父表的分区数目要一致

LOCAL STORE IN (tablespace) (PARTITION [partition [LOGGING|NOLOGGING] [TABLESPACE {tablespace|DEFAULT}] [PCTFREE int] [PCTUSED int] [INITRANS int] [MAXTRANS int] [STORAGE storage_clause] [STORE IN {tablespace_name|DEFAULT] [SUBPARTITION [subpartition [TABLESPACE tablespace]]]])

 

1.3 索引信息存放在父表的分区所在的表空间,这种语法最简单,也是最常用的分区索引创建方式。

Local

1.4 并且指定的Partition 数目要与父表的Partition要一致

LOCAL (PARTITION [partition

[LOGGING|NOLOGGING]

[TABLESPACE {tablespace|DEFAULT}]

[PCTFREE int]

[PCTUSED int]

[INITRANS int]

[MAXTRANS int]

[STORAGE storage_clause]

[STORE IN {tablespace_name|DEFAULT]

[SUBPARTITION [subpartition [TABLESPACE tablespace]]]])

 

Global Index

索引信息的存放位置与父表的Partition信息完全不相干。甚至父表是不是分区表都无所谓的。语法如下:

GLOBAL PARTITION BY RANGE (col_list)

( PARTITION partition VALUES LESS THAN (value_list)

[LOGGING|NOLOGGING]

[TABLESPACE {tablespace|DEFAULT}]

[PCTFREE int]

[PCTUSED int]

[INITRANS int]

[MAXTRANS int]

[STORAGE storage_clause] )

但是在这种情况下,如果父表是分区表,要删除父表的一个分区都必须要更新Global Index ,否则索引信息不正确

ALTER TABLE TableName DROP PARTITION PartitionName Update Global Indexes

 

 

--查询索引

select object_name,object_type,tablespace_name,sum(value)

from v$segment_statistics

where statistic_name IN (‘physical reads‘,‘physical write‘,‘logical reads‘)and object_type=‘INDEX‘

group by object_name,object_type,tablespace_name

order by 4 desc

oracle 分区和分区索引,布布扣,bubuko.com

热门排行

今日推荐

热门手游