Oracle 计算表 + LOBs 大小

Applies To

All Users

Summary

The purpose of this article is to provide more understanding on the space usage of tables when using LOBS.

Solution

A select from the BYTES column in DBA_SEGMENTS for the table shows the table segment but does not include LOB (CLOB or BLOB) segments sizes.

To caluclate the total size for the table and the associated LOBS segments a sum of the following must occur:

the bytes for the table => from dba_segments

the bytes for the LOB segments => from dba_lobs and dba_segments where segment_type is LOBSEGMENT

the bytes for the LOB Index (Lob Locator) = from dba_indexes and dba_segments

The way to accomplish this is to create a SQL file with the following contents (between the lines):

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

ACCEPT SCHEMA PROMPT 'Table Owner: '

ACCEPT TABNAME PROMPT 'Table Name: '

SELECT

(SELECT SUM(S.BYTES) -- The Table Segment size

FROM DBA_SEGMENTS S

WHERE S.OWNER = UPPER('&SCHEMA') AND

(S.SEGMENT_NAME = UPPER('&TABNAME'))) +

(SELECT SUM(S.BYTES) -- The Lob Segment Size

FROM DBA_SEGMENTS S, DBA_LOBS L

WHERE S.OWNER = UPPER('&SCHEMA') AND

(L.SEGMENT_NAME = S.SEGMENT_NAME AND L.TABLE_NAME = UPPER('&TABNAME') AND L.OWNER = UPPER('&SCHEMA'))) +

(SELECT SUM(S.BYTES) -- The Lob Index size

FROM DBA_SEGMENTS S, DBA_INDEXES I

WHERE S.OWNER = UPPER('&SCHEMA') AND

(I.INDEX_NAME = S.SEGMENT_NAME AND I.TABLE_NAME = UPPER('&TABNAME') AND INDEX_TYPE = 'LOB' AND I.OWNER = UPPER('&SCHEMA')))

"TOTAL TABLE SIZE"

FROM DUAL;

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

CASE STUDY (on UNIX)

$vi lob_table_size.sql

ACCEPT SCHEMA PROMPT 'Table Owner: '

ACCEPT TABNAME PROMPT 'Table Name: '

SELECT

(SELECT SUM(S.BYTES) -- The table segment size

FROM DBA_SEGMENTS S

WHERE S.OWNER = UPPER('&SCHEMA') AND

(S.SEGMENT_NAME = UPPER('&TABNAME'))) +

(SELECT SUM(S.BYTES) -- The Lob Segment Size

FROM DBA_SEGMENTS S, DBA_LOBS L

WHERE S.OWNER = UPPER('&SCHEMA') AND

(L.SEGMENT_NAME = S.SEGMENT_NAME AND L.TABLE_NAME = UPPER('&TABNAME') AND L.OWNER = UPPER('&SCHEMA'))) +

(SELECT SUM(S.BYTES) -- The Lob Index size

FROM DBA_SEGMENTS S, DBA_INDEXES I

WHERE S.OWNER = UPPER('&SCHEMA') AND

(I.INDEX_NAME = S.SEGMENT_NAME AND I.TABLE_NAME = UPPER('&TABNAME') AND INDEX_TYPE = 'LOB' AND I.OWNER = UPPER('&SCHEMA')))

"TOTAL TABLE SIZE"

FROM DUAL;

:wq

$ sqlplus / as sysdba;

create user test identified by test;

grant DBA to test;

alter user test default tablespace users;

connect test/test;

-- CREATE THE TABLE FOR THE TEST

CREATE TABLE TEST_TABLE (a number, b CLOB, c CLOB)

LOB(b) STORE AS SECUREFILE (DEDUPLICATE retention none CACHE)

LOB(c) STORE AS SECUREFILE (DEDUPLICATE retention none CACHE);

-- INSERT ROWS INTO THE TABLE

declare

i number;

my_insert_b clob;

my_insert_c clob;

begin

for i in 1..5000 loop

my_insert_b := my_insert_b||'b';

my_insert_c := my_insert_c||'c';

end loop;

for i in 1..1000 loop

insert into TEST_TABLE values (i,to_char(i)||my_insert_b, my_insert_c);

end loop;

commit;

end;

/

-- EXAMINE THE SEGMENTS CREATED BY THE TABLE AND ITS DEPENDENT SEGMENTS

COLUMN SEGMENT_NAME FORMAT A32

COLUMN COLUMN_NAME FORMAT A12

SELECT SEGMENT_NAME, SEGMENT_TYPE, BYTES FROM USER_SEGMENTS;

-- SEGMENT_NAME SEGMENT_TYPE BYTES


-- TEST_TABLE TABLE 262144

-- SYS_IL0000069158C00002$$ LOBINDEX 589824

-- SYS_IL0000069158C00003$$ LOBINDEX 65536

-- SYS_LOB0000069158C00002$$ LOBSEGMENT 19070976

-- SYS_LOB0000069158C00003$$ LOBSEGMENT 1245184

SELECT SUM(BYTES) FROM USER_SEGMENTS;

-- SUM(BYTES)


-- 21233664

-- EXAMINE WHICH LOB SEGMENT BELONGS TO WHICH COLUMN IN TEST_TABLE

SELECT COLUMN_NAME, SEGMENT_NAME

FROM DBA_LOBS

WHERE OWNER = 'TEST' AND TABLE_NAME = 'TEST_TABLE'

-- COLUMN_NAME SEGMENT_NAME


-- B SYS_LOB0000069158C00002$$

-- C SYS_LOB0000069158C00003$$

-- THE LOB INDEXES CAN BE ASSOCIATED WITH A COLUMN BY THE NAME OF THE INDEX WITH REGARD TO THE NAME OF THE LOB SEGMENT

-- EXECUTE THE SCRIPT TO SUM THE LOB STORAGE FOR THE TABLE

run lob_table_size.sql

-- Table Owner: test

-- Table Name: test_table

-- old 4: WHERE S.OWNER = UPPER('&SCHEMA') AND

-- new 4: WHERE S.OWNER = UPPER('test') AND

-- old 5: (S.SEGMENT_NAME = UPPER('&TABNAME'))) +

-- new 5: (S.SEGMENT_NAME = UPPER('test_table'))) +

-- old 8: WHERE S.OWNER = UPPER('&SCHEMA') AND

-- new 8: WHERE S.OWNER = UPPER('test') AND

-- old 9: (L.SEGMENT_NAME = S.SEGMENT_NAME AND L.TABLE_NAME = UPPER('&TABNAME') AND L.OWNER = UPPER('&SCHEMA'))) +

-- new 9: (L.SEGMENT_NAME = S.SEGMENT_NAME AND L.TABLE_NAME = UPPER('test_table') AND L.OWNER = UPPER('test'))) +

-- old 12: WHERE S.OWNER = UPPER('&SCHEMA') AND

-- new 12: WHERE S.OWNER = UPPER('test') AND

-- old 13: (I.INDEX_NAME = S.SEGMENT_NAME AND I.TABLE_NAME = UPPER('&TABNAME') AND INDEX_TYPE = 'LOB' AND I.OWNER = UPPER('&SCHEMA')))

-- new 13: (I.INDEX_NAME = S.SEGMENT_NAME AND I.TABLE_NAME = UPPER('test_table') AND INDEX_TYPE = 'LOB' AND I.OWNER = UPPER('test')))

-- TOTAL TABLE SIZE


-- 21233664

Explanation


LOBs over approximately 4000 bytes in length are stored "out of line" These LOBs are assigned to a special LOB segment that is separate from the table segment.

相关推荐
ClouGence1 天前
Oracle 数据同步为什么会出现数据不一致?长事务是常被忽略的原因
数据库·后端·oracle
ClouGence7 天前
Oracle CDC 架构优化:从主库直连到 DataGuard 备库同步
数据库·后端·oracle
曹牧8 天前
Oracle EXPLAIN PLAN
数据库·oracle
贤时间8 天前
codex 助力oracle ebs 开发
数据库·oracle
秉承初心8 天前
PostgreSQL 数据性能瓶颈突破实战
数据库·postgresql·oracle
Curvatureflight8 天前
MySQL 深分页越来越慢?从 LIMIT OFFSET 改成游标分页
数据库·oracle
XZ-0700018 天前
MySQL事务
数据库·mysql·oracle
tiancaijiben8 天前
阿里云函数计算FC如何实现网站的定时任务与自动化
数据库·oracle·dba
xfhuangfu8 天前
Oracle 19c 多租户体系架构介绍
数据库·oracle·架构
杨云龙UP9 天前
Spotlight 接入 Oracle 数据库监控操作指南 2026-06-16
数据库·oracle·性能监控·预警·阈值·spotlight·瓶颈分析