Oracle的簇与簇表

Oracle的簇与簇表



--========================================

-- 簇表及簇表管理(Index clustered tables)

--========================================

 

    簇表是Oracle中一种可选、的存储表数据的方法。使用簇表可以减少磁盘I/O,改善访问簇表的联结所带来的资源开销,本文讲述了簇表的原理、创建以及管理簇表等。

 

一、什么是簇表及簇表的特性

    1.簇表

         由共享相同数据块的一组表组成。在堆表的管理过程中,对于某些表的某些列和另外的表的某些列经常被用来联结使用,可以将这些表的联结列作为共享的公共列而将这些表组合在一起。这就是簇表形成的原因。例如,scott模式中,有emp表,dept表,两个表经常使用 deptno列来进行联结,为此,我们共享deptno列,将emp和dept表组成簇表。组成簇表后,Oracle物理上将emp和dept表中有关每个部门所有行存储到相同的数据块中。

         簇表不能等同于SQL server中的簇索引,两者并不是一回事。SQL server中的簇索引是使得行的存储按索引键来存储,类似于IOT表。

           

    2.簇键

        簇键是列或多列的组合,为簇表所共有

        在创建簇时指定簇键的列,以后在创建增加的簇中的每个表时,指定相同的列即可

        每个簇键值在簇和簇索引中仅仅存储一次,与不同表中有有多少这样的行无关

       

    3.使用簇表的好处。

        减少磁盘I/O,减少了因使用联结所带来的系统开销

        节省了磁盘存储空间,因为原来需要单独存放多张表,现在可以将联结的部分作为共享列的存储。

   

    4.何时创建簇表

        对于经常查询、当DML较少的表

        表中的记录经常使用到联结查询

       

    5.创建簇表的步骤

        创建簇

        创建簇索引

        创建簇表

       

       

    6.创建簇、簇键、簇表时考虑的问题

        哪些表适用于创建簇

        对于创建簇的表哪些列用作簇列

        创建簇时数据块空间如何使用(pctfree,pctused)

        平均簇键及相关行所需的空间大小

        簇索引的位置(比如存放到不同的表空间)

        预估簇的大小

       

二、创建簇及簇表

 

    在创建簇时,如果未指定索引列,则默认地创建一个索引簇。

    如果指定了散列参数,如hashkeys,hashis 或single table hashkeys,则可以创建散列簇

        SQL> show user;

        USER is "ROBINSON"

        SQL> create cluster emp_dept_cluster(deptno number(2))

          2  pctused 80

          3  pctfree 15

          4  size 1024

          5  tablespace users;

 

        Cluster created.   

 

    在上面创建的簇中,一个最重要的参数就是size,需要为size 指定合适的大小,如果size 指定的太大,则每个块仅仅能存放

    少量的簇,容易引起空间的浪费,如果指定的太小,则容易产生过多的数据链

   

    创建簇索引的条件

        模式中必须包含簇

        必须具有create any index的权限

        簇索引的作用

            用于一个簇键值并返回的包含该簇键值的地址块

        SQL> create index emp_dept_cluster_idx

          2  on cluster emp_dept_cluster;

 

        Index created.

 

    创建簇表

        SQL> create table dept

          2  (deptno number(2) primary key,

          3   dname varchar2(14),

          4   loc   varchar2(13)

          5  )

          6   cluster emp_dept_cluster(deptno);   --使用了cluster关键字后面跟簇名、簇列

 

        Table created.

 

        SQL> create table emp

          2  (empno number primary key,

          3   ename varchar2(10),

          4   job varchar2(9),

          5   mgr number,

          6   hiredate date,

          7   sal number,

          8   comm number,

          9   deptno number(2) references dept(deptno)

         10  )

         11   cluster emp_dept_cluster(deptno);  --使用了cluster关键字后面跟簇名、簇列

 

        Table created.

 

    对于创建的簇表,与普通表的唯一差别是使用了cluster关键字,即告诉oracle 基表的哪一列将映射到簇表中

 

    查看刚刚创建的簇对象

        SQL> select object_name,object_type,status from user_objects order by object_name ;

 

        OBJECT_NAME          OBJECT_TYPE         STATUS

        -------------------- ------------------- -------

        DEPT                 TABLE               VALID     --簇表dept

        EMP                  TABLE               VALID     --簇表emp

        EMP_DEPT_CLUSTER     CLUSTER             VALID     --簇emp_dept_cluster

        EMP_DEPT_CLUSTER_IDX INDEX               VALID     --簇索引

        SYS_C005422          INDEX               VALID

        SYS_C005423          INDEX               VALID

 

   

        SQL> select table_name,tablespace_name,cluster_name,status,pct_free from

          2  dba_tables where owner = 'ROBINSON';

         

        TABLE_NAME      TABLESPACE_NAME CLUSTER_NAME       STATUS     PCT_FREE

        --------------- --------------- ------------------ -------- ----------

        EMP             USERS           EMP_DEPT_CLUSTER   VALID             0

        DEPT            USERS           EMP_DEPT_CLUSTER   VALID             0

 

    下面开始对簇表填充数据

        SQL> begin

          2  for x in ( select * from scott.dept )

          3  loop

          4      insert into dept

          5      values ( x.deptno, x.dname, x.loc );

          6      insert into emp

          7      select * from scott.emp

          8      where deptno = x.deptno;

          9  end loop;

         10  end;

         11  /

 

        PL/SQL procedure successfully completed.

 

三、更改簇

    对于已经创建的簇,我们可以修改簇的相关属性,比如

        修改簇的物理属性(pctfree,pctused,initrans,maxtrans等)

        存储簇键值的所有行所需空间的平均值(size)

        默认的并行度

        alter cluster emp_dpet_cluster

        pctfree 20

        initrans 3;

 

四、删除簇、簇表

    1.删除簇

        可以删除不再需要的簇,删除簇时,簇中对应的表及对应的簇索引都将被删除

        簇数据段占用的盘区以及簇索引段占用的盘区将被释放返还给各自所在的表空间

        删除不包含表及索引的簇

            drop cluster emp_dept_cluster;

         对于包含簇表的簇,可以使用including talbes选项,如果簇中包含表但未使用including tables子句,将收到错误信息

            drop cluster emp_dept_cluster including tables;

        对于包含簇之外的foreign key 约束说参照的主键,需要使用cascade constraints子句

            drop cluster emp_dept including tables cascade constraints

           

    2.删除簇表

        对于不再使用的簇表可以直接使用drop table table_name命令来删除

            drop table emp;

            drop table dept;

   

    3.删除簇索引

        簇索引可以被删除而不影响簇或它的簇表

        若不存在簇索引则簇表也无法使用

        对于簇的访问,则需要重建簇索引

            drop index emp_dept_cluster_idx;

 

五、簇的相关视图

   

    dba_clusters

    all_clusters

    user_clusters

    dba_clu_columns

    user_clu_columns

   

六、演示相关操作

    查看dba_clusters视图获得所创建的簇

        SQL> select cluster_name,tablespace_name,pct_free,pct_used,ini_trans

          2  from dba_clusters where owner = 'ROBINSON';

 

        CLUSTER_NAME         TABLESPACE_NAME                  PCT_FREE   PCT_USED  INI_TRANS

        -------------------- ------------------------------ ---------- ---------- ----------

        EMP_DEPT_CLUSTER     USERS                                 15                     2

 

    查看簇列

        SQL> select * from user_clu_columns;

 

        CLUSTER_NAME         CLU_COLUMN_NAME      TABLE_NAME           TAB_COLUMN_NAME

        -------------------- -------------------- -------------------- ----------------------------------------

        EMP_DEPT_CLUSTER     DEPTNO               DEPT                 DEPTNO

        EMP_DEPT_CLUSTER     DEPTNO               EMP                  DEPTNO

 

    修改簇的相关属性

        SQL> alter cluster emp_dept_cluster

          2  pctfree 20

          3  initrans 3;

 

        Cluster altered.

 

        SQL> select cluster_name,tablespace_name,pct_free,pct_used,ini_trans

          2  from dba_clusters where owner = 'ROBINSON';

 

        CLUSTER_NAME         TABLESPACE_NAME                  PCT_FREE   PCT_USED  INI_TRANS

        -------------------- ------------------------------ ---------- ---------- ----------

        EMP_DEPT_CLUSTER     USERS                                 20                     3

 

    从dba_segments可以看到簇产生了簇段,簇索引产生的为索引段

        SQL> select segment_name,tablespace_name,segment_type from dba_segments where owner = 'ROBINSON';

 

        SEGMENT_NAME         TABLESPACE_NAME                SEGMENT_TYPE

        -------------------- ------------------------------ ------------------

        EMP_DEPT_CLUSTER     USERS                          CLUSTER

        EMP_DEPT_CLUSTER_IDX USERS                          INDEX

        SYS_C005422          USERS                          INDEX

        SYS_C005423          USERS                          INDEX

   

    删除簇,簇为非空时收到错误提示

        SQL> drop cluster emp_dept_cluster; 

        drop cluster emp_dept_cluster

        *

        ERROR at line 1:

        ORA-00951: cluster not empty

 

    使用including tables 删除簇及簇表、簇索引

        SQL> drop cluster emp_dept_cluster including tables;

 

        Cluster dropped.

 

        SQL> select segment_name,tablespace_name from dba_segments where owner = 'ROBINSON';

 

        no rows selected

 



簇由一组共享多个数据块的多个表组成,它将这些表的相关行一起存储到相同数据块中,这样可以减少查询数据所需的磁盘读取量。创建簇后,用户可以在簇中创建表,这些表称为簇表。

例如有如下两个表:student和achievement.。其中,student表存储学生信息,需要使用SID字段(存储学生ID);achievement表存储学生成绩信息,也需要使用SID字段。也就是说,student和achievement需要共享学生ID数据块。

注意:如果用户在自己的模式中创建簇,则必须具有create cluster权限和unlimited tablespace系统权限;如果想在其他模式中创建簇,则还必须具有create any cluster系统权限

创建簇和簇表

创建簇

创建簇,需要使用create cluster语句,例如创建一个名为stu_ach的簇,如下:

[sql] view plain copy

  1. create cluster stu_ach(sid number)  
  2. pctused 40  
  3. pctfree 10  
  4. size 1024  
  5. storage  
  6. (initial 128k  
  7. next 128k  
  8. minextents 2  
  9. maxextents 20  
  10. )tablespace huizhi;  

上面创建簇stu_ach时,指定通过SID字段来对簇中的表进行聚簇存储,这个SID字段就可以称之为聚簇字段。

注意:size子句用来为聚簇字段提供指定的数据块数量。例如,将size设置为1024,即表明簇中的聚簇字段记录只能存储在1024个数据块中。

创建簇表

创建簇表,需要使用cluster子句指定所使用的簇和簇字段。

例如,在stu_ach簇中创建两个簇表:student和achievement。如下:

[sql] view plain copy

  1. create table student(  
  2. sid number,  
  3. sname varchar2(8),  
  4. sage number  
  5. )  
  6. cluster stu_ach(sid);  
  7. --表已创建  
  8. create table achievement(  
  9. aid number,  
  10. score number,  
  11. sid number  
  12. )  
  13. cluster stu_ach(sid);  
  14. --表已创建  

上例在创建student和achievement表时,使用cluster子句指定它们所使用的簇为stu_ach,所使用的簇字段为SID。

提示:将student和achievement两个表组成一个簇后,在物理上oracle会将这两个表中每个学生的学生信息和该学生的所有成绩信息存储到相同的数据块中。

现在向student表中添加记录,如下:

[sql] view plain copy

  1. insert into student values(1,'小明',24);  

发现还无法向簇表中添加记录。

注意:为了能够向簇表中添加记录,还需要首先为簇建立索引。

创建簇索引

簇索引与簇表不同,它并不存在于簇中,而是与普通索引一样需要具有独立的存储空间。

例如,为簇stu_ach建立一个簇索引,如下:

[sql] view plain copy

  1. create index stu_ach_index  
  2. on cluster stu_ach  
  3. tablespace huizhi;  

上例为簇stu_ach建立了一个名为stu_ach_index的簇索引。创建簇索引后,就可以向簇表中添加记录了。

管理簇

对一个簇的管理主要是修改簇和删除簇。如果用户想要管理簇,则必须具有alter any cluster系统权限。

修改簇

修改一个簇,主要是修改簇的如下属性值:

1.物理存储属性,包括pctfree、pctused、initrans、maxtrans和storage。

2.为了存储簇键值的所有行所需空间的平均值size。

3.默认的并行度。

删除簇

1.删除一个空簇:当一个簇中不包含簇表时,删除该簇可以使用drop cluster cluster_name语句。

2.删除一个含有簇表的簇:需要使用drop cluster...including tables语句,如下

[sql] view plain copy

  1. drop cluster stu_ach including tables;  

另外,如果某个簇含有簇表,并且有外键约束,则需要使用drop cluster...including tables cascade constraints语句删除该簇。如下:

[sql] view plain copy

  1. drop cluster stu_ach including tables cascade constraints;  



簇其实就是一组表,由一组共享相同数据块的多个表组成,将经常一起使用的表组合在一起成簇可以提高处理效率;在一个簇中的表就叫做簇表。
建立顺序是:簇→簇表→簇索引→数据
创建簇的格式
CREATE CLUSTER cluster_name
(column date_type [,column datatype]...)
[PCTUSED 40 | integer] [PCTFREE 10 | integer]
[SIZE integer]
[INITRANS 1 | integer] [MAXTRANS 255 | integer]
[TABLESPACE tablespace]
[STORAGE storage]
SIZE:指定估计平均簇键,以及与其相关的行所需的字节数。
1、创建簇


复制代码代码如下:

    create cluster my_clu (deptno number )  
    pctused 60  
    pctfree 10  
    size 1024  
    tablespace users  
    storage (  
    initial 128 k  
    next 128 k  
    minextents 2  
    maxextents 20  
    );  

2、创建簇表


复制代码代码如下:

    create table t1_dept(  
    deptno number ,  
    dname varchar2 ( 20 )  
    )  
    cluster my_clu(deptno);  
    create table t1_emp(  
    empno number ,  
    ename varchar2 ( 20 ),  
    birth_date date ,  
    deptno number  
    )  
    cluster my_clu(deptno);  

3、为簇创建索引


复制代码代码如下:

create index clu_index on cluster my_clu;

注:若不创建簇索引,则在插入数据时报错:ORA-02032: clustered tables cannot be used before the cluster index is built
管理簇
使用ALTER修改簇属性(必须拥有ALTER ANY CLUSTER的权限)
1、修改簇属性
可以修改的簇属性包括:
* PCTFREE、PCTUSED、INITRANS、MAXTRANS、STORAGE
* 为了存储簇键值所有行所需空间的平均值SIZE
* 默认并行度
注:
* 不能修改INITIAL和MINEXTENTS的值
* PCTFREE、PCTUSED、SIZE参数修改后适用于所有数据块
* INITRANS、MAXTRANS仅适用于以后分配的数据块
* STORAGE参数修改后仅影响以后分配给簇的盘区
例:


复制代码代码如下:

    alter cluster my_clu  
    pctused 40  

2、删除簇


复制代码代码如下:

    drop cluster my_clu; -- 仅适用于删除空簇  
    drop cluster my_clu including tables ; -- 删除簇和簇表  
    drop cluster my_clu including tables cascade constraints ;--同时删除外键约束  

注:簇表可以像普通表一样删除。
3、清空簇


复制代码代码如下:

truncate cluster my_clu;

注:所有在此簇上的表的数据全部被清空
散列聚簇表
在簇表中,Oracle使用存储在索引中的键值来定位表中的行,而在散列聚簇表中,使用了散列函数代替了簇索引,先通过内部函数或者自定义的函数进行散列计算,然后再将计算得到的码值用于定位表中的行。创建散列簇需要用到HASHKEYS子句。
1、创建散列簇


复制代码代码如下:

    create cluster my_clu_two(empno number(10) )  
    pctused 70  
    pctfree 10  
    tablespace users  
    hash is empno  
    hashkeys 150 ;  

说明:
* hash is 子句指明了进行散列的列,如果列是唯一的标示行,就可以将列指定为散列值
* hashkeys 指定和限制散列函数可以产生的唯一的散列值的数量
2、创建散列表


复制代码代码如下:

    create table t2_emp (  
    empno number ( 10 ),  
    ename varchar2 ( 20 ),  
    birth_date date ,  
    deptno number )  
    cluster my_clu_two(empno);  

注意:
* 必须设置数值的精度
* 散列簇不能也不用创建索引
* 散列簇不能ALTER:size、hashkeys、hash is参数
不宜用聚簇表的情况
1)如果预料到聚簇中的表会大量修改,聚簇表会对DML的性能产生负面影响
2)非常不适合对单表的全表扫描,因为只能引起对其它表的全表扫描
3)频繁对表进行TRUNCATE和加载,因为聚簇中的表是不能TRUNCATE的,只能TRUNCATE簇
4)如果表只是偶尔被连接或者它们的公共列经常被修改,则不要聚簇表
5)如果经常从所有有相同聚簇键值的表查询出的结果数据超过一个或两个Oracle块,则不要聚簇表
6)如果空间不够,并且不能为将要插入的新记录分配额外的空间,那么不要使用聚簇



About Me


...............................................................................................................................

● 本文整理自网络

● 本文在itpub(http://blog.itpub.net/26736162)、博客园(http://www.cnblogs.com/lhrbest)和个人微信公众号(xiaomaimiaolhr)上有同步更新

● 本文itpub地址:http://blog.itpub.net/26736162/abstract/1/

● 本文博客园地址:http://www.cnblogs.com/lhrbest

● 本文pdf版及小麦苗云盘地址:http://blog.itpub.net/26736162/viewspace-1624453/

● 数据库笔试面试题库及解答:http://blog.itpub.net/26736162/viewspace-2134706/

● QQ群:230161599     微信群:私聊

● 联系我请加QQ好友(646634621),注明添加缘由

● 于 2017-06-02 09:00 ~ 2017-06-30 22:00 在魔都完成

● 文章内容来源于小麦苗的学习笔记,部分整理自网络,若有侵权或不当之处还请谅解

● 版权所有,欢迎分享本文,转载请保留出处

...............................................................................................................................

拿起手机使用微信客户端扫描下边的左边图片来关注小麦苗的微信公众号:xiaomaimiaolhr,扫描右边的二维码加入小麦苗的QQ群,学习最实用的数据库技术。

时间: 2025-01-03 08:00:16

Oracle的簇与簇表的相关文章

Oracle数据库索引聚簇与哈希聚簇使用指南

索引聚簇的使用: ◆对经常在连接语句中访问的表建立聚簇. ◆假如表只是偶尔被连接或者它们的公共列经常被修改,则不要聚簇表.(修改记录的聚簇键值比在 非聚簇的表中修改此值要花费更多的时间,因为Oracle必须将修改的记录移植到其他的块中以维护聚簇) . ◆假如经常需要在一个表上进行完全搜索,则不要聚簇这个表(对一个聚簇表进行完全搜索比在非聚 簇表上进行完全搜索的时间长,Oracle可能要读更多的块,因为表是被一起存储的). ◆假如经常从一个父表和相应的子表中查询记录,则考虑给1对多(1:*)关系创

oracle中用Create Table创建表时,Storage中参数的含义!

oracle|创建 可用于:表空间.回滚段.表.索引.分区.快照.快照日志 参数名称缺省值最小值最大值说明INITIAL5(数据块)2(数据块)操作系统限定分配给Segment的第一个Extent的大小,以字节为单位,这个参数不能在alter语句中改变,如果指定的值小于最小值,则按最小值创建.NEXT5(数据块)1(数据块)操作系统限定第二个Extent的大小等于NEXT的初值,以后的NEXT值=前一NEXT大小乘以(1+PCTINCREASE/100),如果指定的值小于最小值,则按最小值创建.

命令行中使用oracle命令创建数据库和表空间

本文详细介绍命令行中使用oracle命令创建数据库和表空间 1.使用超级用户登录sqlplus. 2.创建表空间: SQL> create tablespace test 2 datafile 'D:\oracle\oradata\test\test.dbf' 3 size 100M autoextend on next 10M maxsize unlimited extent management local; 3.创建用户和密码: SQL> create user Username ide

如何在Oracle中使用临时表解决表冲突

本文介绍一个Oracle里面使用临时表解决表冲突的例子,并附创建临时表的例子 平台:windows 2000 server sp3 数据库:Oracle 9.0.1.1.1 现场描述:用户抱怨不能同时对多个单位进行数据处理,执行出现的错误信息为: SQL> exec sp_hyb_da_ryxx_dwbh_test('331028') BEGIN sp_hyb_da_ryxx_dwbh_test('331028'); END; *ERROR 位于第 1 行: ORA-02055: 分布式更新操作

Oracle学习(一)表空间和区

1.表空间:表空间是数据库的逻辑划分,一个表空间只属于一个数据库.每个表空间由一个或多个 数据文件组成. 表空间中其他逻辑结构的数据存储在这些数据文件中.在Oracle安装完成后,会自动建立多个表空间 , 主要有: (1)System表空间:该表空间用于存放Oracle系统内部表和数据字典的数据,如表名.列名.用户 名等. (2)undo表空间:该表空间是存储撤销信息的表空间.当用户对数据库表进行修改(insert, update,delete)时, Oracle会自动使用undo表空间来临时存

oracle中如何判断本表有多少张表对其有外键关联

在oracle中如何判断本表有多少张表对其有外键关联?我们主要分为两步: 第一步:        //查询表的主键约束名 select * from user_constraints e where e.table_name='' -----此处输入表名 第二步:        //查询所有引用到该主键的记录 select b.table_name,b.column_name from user_constraints a inner join user_cons_columns b on a.

如何在oracle 11g 中导出空表

  如何在oracle 11g 中导出空表 由于oracle 11g的 延迟段创建的新特性,导致在没有数据插入时,oracle是不会分配数据段的,进而导致exp 是不能导出11g数据库的空表的. 当然采用expdp就不存在这个问题了. expdp hr/hr schemas=hr dumpfile=expdp.dmp directory=dbtest conn hr/hr select TABLE_NAME,NUM_ROWS from user_tables; TABLE_NAME NUM_RO

探索ORACLE之RMAN_07整个业务表空间丢失恢复

探索ORACLE之RMAN_07整个业务表空间丢失恢复 作者:吴伟龙   Name:Prodence Woo QQ:286507175  msn:hapy-wuweilong@hotmail.com   1.     整个业务表空间丢失恢复 注意:以下的所有实验,都是基于上面的全库备份来做的恢复. 2.1 删除wwl表空间的所有数据文件 [root@wwldb ~]# cd /DBData/WWL/ [root@wwldb WWL]# rm -rf wwl* [root@wwldb WWL]#

oracle表记录导入导出-oracle数据库的几张表的记录导入导出

问题描述 oracle数据库的几张表的记录导入导出 在项目组,为了实现把oracle数据库中几张表的数据从一个生产环境导出再导入另一个生产环境,怎样通过前台页面来实现导入导出功能? 解决方案 前台页面来写的话,就得在后台来实现一个将数据导出到文本的接口(查全部,导入成txt.或excel) 同时,还要实现一个从文本直接加载到数据库的接口(或读文件,批量插入)