最近有朋友问到了DBMS_SPACE包的使用,也看了一下,大部分是关于dbms_space.space_usage的使用,space_usage这个过程的例子已经很多了,我也就不再多说了,除了这个过程外,另外还有两个过程也有着特殊的用处,但使用的人不多,我们也来看看这两个过程有什么用。

这两个过程为:CREATE_INDEX_COST和CREATE_TABLE_COST,分别用户评估创建索引和创建表的存储开销(空间占用情况)。

CREATE_INDEX_COST的语法如下:

DBMS_SPACE.CREATE_INDEX_COST (

ddl            IN   VARCHAR2,

used_bytes     OUT  NUMBER,

alloc_bytes    OUT  NUMBER,

plan_table     IN   VARCHAR2 DEFAULT NULL);

下面是相关的测试代码

1、准备相关表和数据

SQL> set serveroutput on

SQL> create table t(c char(100),d varchar2(200));

表已创建。

SQL> begin

2   for i in 1..5000 loop

3    insert into t values(i,i);

4   end loop;

5   commit;

6 end;

7 /

PL/SQL过程已成功完成。

2、分析表,注意:没有统计信息,CREATE_INDEX_COST将无法计算索引的存储开销

SQL> analyze table t compute statistics;

表已分析。

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(c)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

488.28125 640   --计算出的索引将占用488K字节空间,为该索引需要分配640k存储空间

PL/SQL过程已成功完成。

3、创建实际索引,确定索引存储空间是否与计算的结果相符

SQL> create index i on t(c);

索引已创建。

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

11

已选择1行。 --11个64k的区,比计算出的大1个区

4、再次装载数据

SQL> begin

2   for i in 1..5000 loop

3    insert into t values(i,i);

4   end loop;

5   commit;

6 end;

7 /

PL/SQL过程已成功完成。

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(c)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

488.28125 640          --没有分析之前,获得得仍然是根据以前分析结果计算的值

PL/SQL过程已成功完成。

SQL> analyze table t compute statistics;

表已分析。

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(c)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

976.5625 2048                 --分析之后,得到新的结果

PL/SQL过程已成功完成。

5、再次验证,16个64k的区和1个1024k的区,2048k,与估计值相同

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

17

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

6、换了一个字段进行测试

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(d)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

39.0625 192       --计算出的索引将占用39K字节空间,为该索引需要分配192k存储空间

PL/SQL过程已成功完成。

7、创建索引,新建的索引比估算的值大1个区

SQL> create index i on t(d);

索引已创建。

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

4

SQL> drop index i;

8、再次装载数据并分析表

SQL> begin

2   for i in 1..10000 loop

3    insert into t values(i,i);

4   end loop;

5   commit;

6 end;

7 /

PL/SQL过程已成功完成。

SQL> analyze table t compute statistics;

表已分析。

9、重新计算,得到新的估算值

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(d)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

78.125 320

PL/SQL过程已成功完成。

10、创建索引,新建的索引比估计的大2个区

SQL> create index i on t(d);

索引已创建。

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

7

11、顺便测试shink space的效果

SQL> select count(*) from t;

COUNT(*)

----------

20000

SQL> delete t where rownum<=15000;

已删除15000行。

SQL> commit;

提交完成。

SQL> alter table t enable row movement;

表已更改。

12、在删掉15000行数据后,没有整理空间之前进行统计信息收集

SQL> analyze table t compute statistics;

表已分析。

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(d)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

24.4140625 128         --基于新收集的统计信息计算,估算的索引需要分配128k存储空间

PL/SQL过程已成功完成。

13、收缩表,释放占用的存储空间

SQL> alter table t shrink space;

表已更改。

SQL> analyze table t compute statistics;

表已分析。

SQL> declare

2   v1 number;

3   v2 number;

4 begin

5   DBMS_SPACE.CREATE_INDEX_COST('create index i on t(d)',v1,v2);

6   dbms_output.put_line(v1/1024||' '||v2/1024);

7 end;

8 /

24.4140625 128  --收缩后重新收集统计信息,与原统计信息一样,因此计算出的大小一样

PL/SQL过程已成功完成。

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

7

--现有索引并没有收缩,仅仅是表空间进行了收缩,因此现有索引仍保持原大小

14、重建索引,对比新的索引大小与计算出的索引大小一样大

SQL> alter index i rebuild;

索引已更改。

SQL> select count(*) from user_extents where segment_name='I';

COUNT(*)

----------

2

--重建索引后新的索引占用空间与计算出的空间一样大

CREATE_TABLE_COST有两种用法,因此包内进行了overload,具体的语法如下:

DBMS_SPACE.CREATE_TABLE_COST (

tablespace_name   IN VARCHAR2,

avg_row_size      IN NUMBER,

row_count         IN NUMBER,

pct_free          IN NUMBER,

used_bytes        OUT NUMBER,

alloc_bytes       OUT NUMBER);

DBMS_SPACE.CREATE_TABLE_COST (

tablespace_name   IN VARCHAR2,

colinfos          IN CREATE_TABLE_COST_COLUMNS,

row_count         IN NUMBER,

pct_free          IN NUMBER,

used_bytes        OUT NUMBER,

alloc_bytes       OUT NUMBER);

CREATE TYPE create_table_cost_colinfo IS OBJECT (

COL_TYPE  VARCHAR(200),

COL_SIZE  NUMBER);

下面是关于CREATE_TABLE_COST的测试代码

1、测试创建一个表所需的存储大小,预计该表平均行长度为100字节,10000行数据

SQL> DECLARE

2   V1 NUMBER;

3   V2 NUMBER;

4 BEGIN

5   DBMS_SPACE.CREATE_TABLE_COST('USERS', 100, 10000, 10, V1, V2);

6   DBMS_OUTPUT.PUT_LINE('V1: '||V1/1024/8||' V2: '||V2/1024/8);

7 END;

8 /

V1: 143 V2: 256         --估算出该表需要存储空间143块,所需分配空间256块

PL/SQL过程已成功完成。

2、创建该表,并插入10000行数据

SQL> CREATE TABLE T1(C CHAR(96));         --96字节的char字段平均行长度为100字节

表已创建。

SQL> BEGIN

2   FOR I IN 1..10000 LOOP

3    INSERT INTO T1 VALUES(I);

4   ENDLOOP;

5   COMMIT;

6 END;

7 /

PL/SQL过程已成功完成。

3、分析表统计信息

SQL> ANALYZE TABLE T1 COMPUTE STATISTICS;

表已分析。

SQL> SELECT BLOCKS,EMPTY_BLOCKS,AVG_ROW_LEN

FROM USER_TABLES WHERE TABLE_NAME='T1';

BLOCKS EMPTY_BLOCKS AVG_ROW_LEN

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

180          76        100

--经检查,高水平线之前的块数180块,高水平线之后的空块数76块,总存储空间为256块,与DBMS_SPACE.CREATE_INDEX_COST计算出的总需要存储空间大小相符。

4、通过dbms_space.space_usage过程,可以进一步看到表中各个块的使用情况

declare

unf number;

unfb number;

fs1 number;

fs1b number;

fs2 number;

fs2b number;

fs3 number;

fs3b number;

fs4 number;

fs4b number;

full number;

fullb number;

own dba_tables.owner%type;

tab dba_tables.table_name%type;

yesno varchar2(3);

type parts is table of dba_tab_partitions%rowtype;

partlist parts;

type cursor_ref is ref cursor;

c_cur cursor_ref;

begin

own:=upper('&owner');

tab:=upper('&table_name');

dbms_output.put_line('--------------------------------------------------------------------------------');

open c_cur for select partitioned from dba_tables

where wner=own and table_name=tab;

fetch c_cur into yesno;

close c_cur;

dbms_output.put_line('Owner:    '||own);

dbms_output.put_line('Table:    '||tab);

dbms_output.put_line('------------------------------------------------');

if yesno='NO' then

dbms_space.space_usage(own,tab,'TABLE',unf,unfb,fs1,fs1b,fs2,fs2b,fs3,fs3b,fs4,fs4b,full,fullb);

dbms_output.put_line('unf: '||unf||' fs1: '||fs1||' fs2: '||fs2||' fs3: '||fs3||' fs4: '||fs4||' full: '||full);

else

open c_cur for select * from dba_tab_partitions

where table_owner=own and table_name=tab;

fetch c_cur bulk collect into partlist;

close c_cur;

for i in partlist.first .. partlist.last   loop

dbms_space.space_usage(partlist(i).table_owner,partlist(i).table_name,'TABLE PARTITION',unf,unfb,fs1,fs1b,fs2,fs2b,fs3,fs3b,fs4,fs4b,full,fullb,partlist(i).partition_name);

dbms_output.put_line('Partition: '||partlist(i).partition_name);

dbms_output.put_line('unf: '||unf||' fs1: '||fs1||' fs2: '||fs2||' fs3: '||fs3||' fs4: '||fs4||' full: '||full);

end loop;

end if;

dbms_output.put_line('--------------------------------------------------------------------------------');

end;

/

输入owner的值: HR

原值  22:  own:=upper('&owner');

新值  22:  own:=upper('HR');

输入table_name的值: T1

原值  23:  tab:=upper('&table_name');

新值  23:  tab:=upper('T1');

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

Owner:    HR

Table:    T1

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

unf: 0 fs1: 1 fs2: 0 fs3: 0 fs4: 39 full: 140

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

PL/SQL过程已成功完成。

--经查看,发现该表写满数据的块有140块,3/4满的块有39块,1/4满的块有1块,该表存储空间没有有效利用,可以看到140+39+1=180,这些均为高水平线之下的块。但与DBMS_SPACE.CREATE_INDEX_COST计算出的数据需要143块不符。

5、对表进行空间整理并重新分析

SQL> ALTER TABLE T1 MOVE;

表已更改。

SQL> ANALYZE TABLE T1 COMPUTE STATISTICS;

表已分析。

SQL> SELECT BLOCKS,EMPTY_BLOCKS,AVG_ROW_LEN

FROM USER_TABLES WHERE TABLE_NAME='T1';

BLOCKS EMPTY_BLOCKS AVG_ROW_LEN

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

155         101        100

--经检查,高水平线之前的块数155,高水平线之后的空块数101,平均行长度100字节

6、通过dbms_space.space_usage过程,可以进一步看到表中各个块的使用情况

declare

unf number;

unfb number;

fs1 number;

fs1b number;

fs2 number;

fs2b number;

fs3 number;

fs3b number;

fs4 number;

fs4b number;

full number;

fullb number;

own dba_tables.owner%type;

tab dba_tables.table_name%type;

yesno varchar2(3);

type parts is table of dba_tab_partitions%rowtype;

partlist parts;

type cursor_ref is ref cursor;

c_cur cursor_ref;

begin

own:=upper('&owner');

tab:=upper('&table_name');

dbms_output.put_line('--------------------------------------------------------------------------------');

open c_cur for select partitioned from dba_tables

where wner=own and table_name=tab;

fetch c_cur into yesno;

close c_cur;

dbms_output.put_line('Owner:    '||own);

dbms_output.put_line('Table:    '||tab);

dbms_output.put_line('------------------------------------------------');

if yesno='NO' then

dbms_space.space_usage(own,tab,'TABLE',unf,unfb,fs1,fs1b,fs2,fs2b,fs3,fs3b,fs4,fs4b,full,fullb);

dbms_output.put_line('unf: '||unf||' fs1: '||fs1||' fs2: '||fs2||' fs3: '||fs3||' fs4: '||fs4||' full: '||full);

else

open c_cur for select * from dba_tab_partitions

where table_owner=own and table_name=tab;

fetch c_cur bulk collect into partlist;

close c_cur;

for i in partlist.first .. partlist.last   loop

dbms_space.space_usage(partlist(i).table_owner,partlist(i).table_name,'TABLE PARTITION',unf,unfb,fs1,fs1b,fs2,fs2b,fs3,fs3b,fs4,fs4b,full,fullb,partlist(i).partition_name);

dbms_output.put_line('Partition: '||partlist(i).partition_name);

dbms_output.put_line('unf: '||unf||' fs1: '||fs1||' fs2: '||fs2||' fs3: '||fs3||' fs4: '||fs4||' full: '||full);

end loop;

end if;

dbms_output.put_line('--------------------------------------------------------------------------------');

end;

/

输入owner的值: HR

原值  22:  own:=upper('&owner');

新值  22:  own:=upper('HR');

输入table_name的值: T1

原值  23:  tab:=upper('&table_name');

新值  23:  tab:=upper('T1');

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

Owner:    HR

Table:    T1

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

unf: 0 fs1: 0 fs2: 0 fs3: 0 fs4: 0 full: 143

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

PL/SQL过程已成功完成。

--经查看,发现该表写满数据的块有143块,与DBMS_SPACE.CREATE_INDEX_COST计算出的数据需要块数完全相同

-- review the parameters

SELECT argument_name, data_type, type_owner, type_name

FROM all_arguments

WHERE object_name = 'CREATE_TABLE_COST'

AND verload = 2

-- examine the input parameter type

SELECT text

FROM dba_source

WHERE name = 'CREATE_TABLE_COST_COLUMNS';

-- drill down further into the input parameter type

SELECT text

FROM dba_source

WHERE name = 'create_table_cost_colinfo';

set serveroutput on

DECLARE

ub NUMBER;

ab NUMBER;

cl sys.create_table_cost_columns;

BEGIN

cl := sys.create_table_cost_columns( sys.create_table_cost_colinfo('NUMBER',10),

sys.create_table_cost_colinfo('VARCHAR2',30),

sys.create_table_cost_colinfo('VARCHAR2',30),

sys.create_table_cost_colinfo('DATE',NULL));

DBMS_SPACE.CREATE_TABLE_COST('SYSTEM',cl,100000,0,ub,ab);

DBMS_OUTPUT.PUT_LINE('Used Bytes: ' || TO_CHAR(ub));

DBMS_OUTPUT.PUT_LINE('Alloc Bytes: ' || TO_CHAR(ab));

END;

/

DBMS_SPACE包的使用相关推荐

  1. 深入SecureFile—新一代LOB揭秘000

    编写人:常伟俊 审核人:张陈亚 在Oracle8i 中,LOB 设计是基于下列假设做出的: (1) LOB 实例化的大小预计为几个兆字节. (2) LOB 通常被视为"单写多读"类 ...

  2. SQL与Oracle对比

    关键字: SQL,ORACLE 分类: 个人专区 密级: 公开 (评分:★★★★ , 回复: 16, 阅读: 641) »» 本人平时比较沉默,但朋友们都说我很疯狂-疯狂地学习.疯狂地工作.很久就 ...

  3. Oracle如何预估将要创建的索引和表的大小

    Oracle如何预估将要创建的索引和表的大小 1. 对于表和索引空间的预估,可以使用DBMS_SPACE包的CREATE_TABLE_COST和CREATE_INDEX_COST存储过程,虽然没有看这 ...

  4. Linux常用命令全集

    Table Of Contents 1. Linux常用命令全集.CHM 9 1.1 备份压缩 9 1.1.1 ar 9 1.1.2 bunzip2 11 1.1.3 bzip2 12 1.1.4 b ...

  5. 28 Oracle深度学习笔记——ORACLE自带DBMS函数包

    28.Oracle深度学习笔记--ORACLE自带DBMS函数包 欢迎转载,转载请标明出处:http://blog.csdn.net/notbaron/article/details/50830889 ...

  6. 28.Oracle深度学习笔记——ORACLE自带DBMS函数包

    28.Oracle深度学习笔记--ORACLE自带DBMS函数包 欢迎转载,转载请标明出处:http://blog.csdn.net/notbaron/article/details/50830889 ...

  7. IDEA设置单个文件、单个包、单个项目的编码格式

    IDEA设置单个文件.单个包.单个项目的编码格式 File-> Settings-> File Enclodings 选择编码格式,确定即可. 注意:此处的编码格式设定以后,该包已经存在的 ...

  8. JPA不同包下同类名查询出错

    不同包下同类名查询出错 异常现象:使用JPA进行查询时,JPA的实体类映射到了另外一个包下的同名类,由于两个同名类中字段名和字段数不相同,所以会出现种种查询错误. 原因及对策:这里应该是jpa实体类默 ...

  9. 手动将web项目的class文件打成jar包,手动打jar包,java -cvf,IDE打包底层指令

    手动将web项目的class文件打成jar包. 我们的项目在使用IDE进行编译后,在项目的target目录下将会生成class文件.我们可以将class文件打成jar包. 使用的到命令为: 在targ ...

最新文章

  1. 无屏幕和键盘配置树莓派WiFi和SSH
  2. 利用html5标签audio在不同客户端下适配播放音频
  3. php$上传_如何实现PHP上传视频的功能?(图文+视频)
  4. 打造具备互补测试技能的团队
  5. 昆仑网(D×××)去中心化虚拟互联网基础功能图片式介绍,请全屏看。
  6. python 回车字符_「答案」python每日一题20201107
  7. 玩转mini2440开发板之【64位Ubuntu系统无法运行arm-linux-gcc】
  8. 对象运算符.和[]的用法
  9. 一文入门基于三维数据的深度学习
  10. 滴滴顺风车再调整服务时间:男女用户一视同仁
  11. 基于JAVA+SpringMVC+MYSQL的医院后勤管理系统
  12. 简明python教程-简明Python教程-中文版.pdf
  13. unity设置一个物体是另一个物体的子物体
  14. java.exe和javaw.exe有什么区别吗?
  15. 读书 | 如何像沉迷游戏一样对工作上瘾?
  16. ROS Spinning-----简介
  17. 用emWin的2D绘图函数画一个带圆形端点的环形进度条
  18. 电脑进不去游戏显示重新连接服务器,幻塔无法连接服务器怎么办?游戏进不去解决方法...
  19. 统计之 - 独立性检验
  20. Kubernetes pod 状态出现 ImagePullBackOff 的原因

热门文章

  1. tensorflow学习:分布式tensorflow使用,代码实现inGraph方式
  2. 使用client-go自定义开发Kubernetes
  3. Python代码打包成exe文件
  4. 缺少动态连接库.so--cannot open shared object file: No such file or directory
  5. c++ STL 容器——序列
  6. UOJ58 【WC2013】糖果公园
  7. 自制操作系统(十) 图像叠加处理
  8. Android Studio开发第二篇创建新项目
  9. drf2 FBV和CBV
  10. S1 Python 基础