Oracle高水位線(HWM)及性能優(yōu)化

ORACLE在邏輯存儲(chǔ)上分4個(gè)粒度:表空間,段,區(qū)和塊.
  • 塊:是粒度最小的存儲(chǔ)單位,現(xiàn)在標(biāo)準(zhǔn)的塊大小是8K,ORACLE每一次I/O操作也是按塊來操作的,也就是說當(dāng)ORACLE從數(shù)據(jù)文件讀數(shù)據(jù)時(shí),是讀取多少個(gè)塊,而不是多少行.
  • 區(qū):由一系列相鄰的塊而組成,這也是ORACLE空間分配的基本單位,舉個(gè)例子來說,當(dāng)我們創(chuàng)建一個(gè)表PM_USER時(shí),首先ORACLE會(huì)分配一區(qū)的空間給這個(gè)表,隨著不斷的INSERT數(shù)據(jù)到PM_USER,原來的這個(gè)區(qū)容不下插入的數(shù)據(jù)時(shí),ORACLE是以區(qū)為單位進(jìn)行擴(kuò)展的,也就是說再分配多少個(gè)區(qū)給PM_USER,而不是多少個(gè)塊.
  • 段:是由一系列的區(qū)所組成,一般來說,當(dāng)創(chuàng)建一個(gè)對(duì)象時(shí)(表,索引),就會(huì)分配一個(gè)段給這個(gè)對(duì)象.所以從某種意義上來說,段就是某種特定的數(shù)據(jù).如CREATE TABLE PM_USER,這個(gè)段就是數(shù)據(jù)段,而CREATE INDEX ON PM_USER(NAME),ORACLE同樣會(huì)分配一個(gè)段給這個(gè)索引,但這是一個(gè)索引段了.查詢段的信息可以通過數(shù)據(jù)字典: SELECT * FROM USER_SEGMENTS來獲得,
  • 表空間:包含段,區(qū)及塊.表空間的數(shù)據(jù)物理上儲(chǔ)存在其所在的數(shù)據(jù)文件中.一個(gè)數(shù)據(jù)庫至少要有一個(gè)表空間.
確定碎片程度
  SELECT table_name,
         ROUND ( (blocks * 8), 2) "高水位空間 k",
         ROUND ( (num_rows * avg_row_len / 1024), 2) "真實(shí)使用空間 k",
         ROUND ( (blocks * 10 / 100) * 8, 2) "預(yù)留空間(pctfree) k",
         ROUND (
            (  blocks * 8
             - (num_rows * avg_row_len / 1024)
             - blocks * 8 * 10 / 100),
            2)
            "浪費(fèi)空間 k"
    FROM user_tables
   WHERE temporary = 'N'
ORDER BY 5 DESC;
查看表上次收集統(tǒng)計(jì)信息時(shí)間
 select table_name,last_analyzed from dba_tables where owner = 'SCHEMA_NAME';
  • 整理表,不影響DML操作
SQL> alter table TABLE_NAME enable ROW MOVEMENT;--啟動(dòng)行移動(dòng)功能
SQL> alter table TABLE_NAME shrink space compact;  --只整理碎片 不回收空間
SQL> alter table TABLE_NAME disable ROW MOVEMENT;--關(guān)閉行移動(dòng)

如果要同時(shí)壓縮表的索引,可以發(fā)布:ALTER TABLE TEST_TAB SHRINK SPACE CASCADE

  • 重置高水位,此時(shí)不能有DML操作
SQL> alter table TABLE_NAME enable ROW MOVEMENT;--啟動(dòng)行移動(dòng)功能
SQL> alter table TABLE_NAME shrink space; --整理碎片并回收空間,并調(diào)整水位線。業(yè)務(wù)少時(shí)執(zhí)行
SQL> alter table TABLE_NAME disable ROW MOVEMENT;--關(guān)閉行移動(dòng)
  • shrink 的優(yōu)勢(shì):
    不需要重建索引。
    可以在線操作
釋放表的高水位通常有如下幾種辦法:
  • 對(duì)表進(jìn)行MOVE操作:ALTER TABLE TABLE_NAME MOVE;。若表上存在索引,則記得重建索引。
  • 對(duì)表進(jìn)行SHRINK SPACE操作:ALTER TABLE TABLE_NAME SHRINK SPACE;,注意,在執(zhí)行該指令之前必須開啟行移動(dòng):ALTER TABLE TABLE_NAME ENABLE ROW MOVEMENT;。該方法的優(yōu)點(diǎn)是:在碎片整理結(jié)束后,表上相關(guān)的索引仍然有效,缺點(diǎn)是會(huì)產(chǎn)生大量的UNDO和REDO。
  • 復(fù)制要保留的數(shù)據(jù)到臨時(shí)表T,DROP原表,然后RENAME臨時(shí)表T為原表。
  • exp/imp或expdp/impdp重構(gòu)表。
  • 若表中沒有數(shù)據(jù)則直接使用TRUNCATE來釋放高水位。
如何找出系統(tǒng)中哪些表擁有高水位呢?
  • 比較表的行數(shù)和表的大小關(guān)系。如果行數(shù)為0,而表的當(dāng)前占用大小減去初始化時(shí)的大?。↖NITIAL_EXTENT)后依然很大,那么說明該表有高水位。
  • 行數(shù)和塊數(shù)的比率,即查看一個(gè)塊可以存儲(chǔ)多少行數(shù)據(jù)。如果一個(gè)塊存儲(chǔ)的行數(shù)少于5行甚至更少,那么說明有高水位。
    注意:
    這兩種方法都不是十分準(zhǔn)確,需要再對(duì)查詢結(jié)果進(jìn)行篩選。需要注意的是,在查詢表的高水位時(shí),首先需要分析表,以得到最準(zhǔn)確的統(tǒng)計(jì)信息。
SELECT OWNER,

       SEGMENT_NAME TABLE_NAME,

       SEGMENT_TYPE,

       GREATEST(ROUND(100 * (NVL(HWM - AVG_USED_BLOCKS, 0) /

                      GREATEST(NVL(HWM, 1), 1)),

                      2),

                0) WASTE_PER

  FROM (SELECT A.OWNER OWNER,

               A.SEGMENT_NAME,

               A.SEGMENT_TYPE,

               B.LAST_ANALYZED,

               A.BYTES,

               B.NUM_ROWS,

               A.BLOCKS BLOCKS,

               B.EMPTY_BLOCKS EMPTY_BLOCKS,

               A.BLOCKS - B.EMPTY_BLOCKS - 1 HWM,

               DECODE(ROUND((B.AVG_ROW_LEN * NUM_ROWS *

                            (1 + (PCT_FREE / 100))) / C.BLOCKSIZE,

                            0),

                      0,

                      1,

                      ROUND((B.AVG_ROW_LEN * NUM_ROWS *

                            (1 + (PCT_FREE / 100))) / C.BLOCKSIZE,

                            0)) + 2 AVG_USED_BLOCKS,

               ROUND(100 *

                     (NVL(B.CHAIN_CNT, 0) / GREATEST(NVL(B.NUM_ROWS, 1), 1)),

                     2) CHAIN_PER,

               B.TABLESPACE_NAME O_TABLESPACE_NAME

          FROM SYS.DBA_SEGMENTS A, SYS.DBA_TABLES B, SYS.TS$ C

         WHERE A.OWNER = B.OWNER

           AND SEGMENT_NAME = TABLE_NAME

           AND SEGMENT_TYPE = 'TABLE'

           AND B.TABLESPACE_NAME = C.NAME)

 WHERE GREATEST(ROUND(100 * (NVL(HWM - AVG_USED_BLOCKS, 0) /

                      GREATEST(NVL(HWM, 1), 1)),

                      2),

                0) > 50

   AND OWNER NOT LIKE '%SYS%'

   AND BLOCKS > 100

 ORDER BY WASTE_PER DESC;
SELECT D.OWNER,

       ROUND(D.NUM_ROWS / D.BLOCKS, 2),

       D.NUM_ROWS,

       D.BLOCKS,

       D.TABLE_NAME,

 ROUND((d.BLOCKS*8-D.INITIAL_EXTENT/1024)/1024)  t_size

  FROM DBA_TABLES D

 WHERE D.BLOCKS > 10

   AND ROUND(D.NUM_ROWS / D.BLOCKS, 2) < 5

 AND d.OWNER NOT LIKE '%SYS%' ;
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容