實例對比Oracle中truncate和delete的區別

刪除表中的數據的方法有delete,truncate,
它們都是刪除表中的數據,而不能刪除表結構,delete 能夠刪除整個表的數據也能夠刪除表中某一條或N條知足條件的數據,truncate只能刪除整個表的數據,通常咱們把delete 操做收做刪除表,truncate操做叫做截斷表.
truncate 操做與 delete 操做對比
操做
 回滾 
 高水線 
 空間
 效率 
Truncate
 不能
 降低
 回收
delete
 能夠
 不變
 不回收
 
下面分別用實例查看它們的不一樣

1.回滾

首先要明白兩點
1.oracle 中數據刪除後還能回滾是由於它把原始數據放到了undo表空間,
2.DML語句使用undo表空間,DDL語句不使用undo,deleteDML語句,truncateDDL語句,別外DDL語句是隱式提交.
因此truncate操用不能回滾,delete操做能夠.
兩種操做對比(首先新建一個表,並插入數據)
SQL> create table t
  2  (
  3  i number
  4  );
Table created.
SQL> insert into t values(10);
SQL> commit;
Commit complete.
SQL> select * from t;
         I
----------
        10


Delete刪除,而後回滾html

SQL> delete from t;
1 row deleted.
SQL> select * from t;
no rows selected
# 刪 除 後回 滾
SQL> rollback;
Rollback complete.
SQL> select * from t;
         I
----------
        10

Truncate截斷表,而後回滾.linux

SQL> truncate table t;
Table truncated.
SQL> rollback;
Rollback complete.
SQL> select * from t;
no rows selected

可見delete刪除表還能夠回滾,truncate截斷表就不能回滾了.(前提是delete操做沒有提交)

2.高水線

全部的Oracle表都有一個容納數據的上限(很象一個水庫歷史最高的水位),咱們把這個上限稱爲「high water mark」HWM。這個HWM是一個標記(專門有一個數據塊用來記錄高水標記等),用來講明已經有多少數據塊分配給這個表. HWM一般增加的幅度爲一次5個數據塊.
delete語句不影響表所佔用的數據塊高水線(high watermark)保持原位置不動
truncate 語句缺省狀況下空間釋放,除非使用reuse storage;   truncate會將高水線復位
下面對兩種操做對比

SQL> analyze table t estimate statistics;
Table analyzed.
SQL> select segment_name,blocks from dba_segments where segment_name=upper('t');
SEGMENT_NAME                       BLOCKS
------------------------------ ----------
T                                      24
SQL> select table_name,blocks,empty_blocks from user_tables where table_name=upper('t');
TABLE_NAME                         BLOCKS EMPTY_BLOCKS
------------------------------ ---------- ------------
T                                      20            3

USER_TABLES.BLOCKS 列表明該表中曾經使用過得數據庫塊的數目,即水線。sql

注意:USER_TABLES.BLOCKS EMPTY_BLOCKS (20+3=23)DBA_SEGMENTS.BLOCKS少一個數據庫塊,這是由於有一個數據庫塊被保留用做表頭。DBA_SEGMENTS.BLOCKS 表示分配給這個表的全部的數據庫塊的數目。USER_TABLES.BLOCKS表示已經使用過的數據庫塊的數目(水線)
Delete刪除表,
SQL> delete from t;
10000 rows deleted
SQL> commit;
Commit complete.
SQL> analyze table t estimate statistics;
Table analyzed.
SQL> select table_name,blocks,empty_blocks from user_tables where table_name=upper('t');
TABLE_NAME                         BLOCKS EMPTY_BLOCKS
------------------------------ ---------- ----------------------------------------------------------------
T                                      20            3

Truncate
截斷表
SQL> truncate table t;
Table truncated.
SQL> analyze table t estimate statistics;
Table analyzed.
SQL> select table_name,blocks,empty_blocks from user_tables where table_name=upper('t');
TABLE_NAME                         BLOCKS EMPTY_BLOCKS
------------------------------ ---------- --------------------------------------------------------
T                                                   7

可見
,delete,BLOCK(高水線)不變,truncateBLOCKS(高水線)變爲
如今咱們也看到blocks+empty_blocks=7,也就是oracle分配區時默認一次7+1(表頭)=8blocks;
高水線的做用: HWM數據庫的操做有以下影響:
a) 全表掃描一般要讀出直到HWM標記的全部的屬於該表 數據庫 塊,即便該表中沒有任何數據。
b) 即便HWM如下有空閒的數據庫塊,鍵入在插入數據時使用了append關鍵字,則在插入時使用HWM以上的數據塊,此時HWM會自動增大。
所以高水線是oracle優化時一個重要的參數

3.空間

既然高水線用來講明已經有多少數據塊分配給這個表,那麼高水線也可理解爲表的空間佔用。
即便delete將表中的數據所有刪除,HWM仍是爲原值,因此還有那麼多的空間分配給這個表,即它的空間尚未回收,
truncate表後高水線變爲,那如今它就表示沒有分配空間,即它的空間被回收了。

4.效率

要想查看delete,truncate那個效率更高,先構建一個大表,而後查看它們分別對些表刪除所需的時間。
有個至關形象的比喻:領導給你兩本書讓你扔掉,delete就是你守在複印機前,把書一頁頁撕下來複印一份,再一頁頁扔到垃圾桶裏,truncate就是直接把兩本書扔到垃圾桶裏,那個快那個慢不言而喻。
先在表中插入100000條記錄,並打開時間
SQL> set timing on;
SQL> begin
  2  for i in 1..100000 loop
  3  insert into t values('10');
  4  commit;
  5  end loop;
  6  end;
  7  /
PL/SQL procedure successfully completed.
Elapsed: 00:01:12.50

Delete
刪除表
SQL> delete from t;
100000 rows deleted.
Elapsed: 00:00:20.09

Truncate 
截斷表
# 先把表回滾
SQL> rollback;
Rollback complete.
Elapsed: 00:00:17.36
SQL> select count(*) from t;
  COUNT(*)
-------------------
    100000
Elapsed: 00:00:00.01
SQL> truncate table t;
Table truncated.
Elapsed: 00:00:00.20

可見刪除同一個大小的表,
delete用了20.09秒,而truncate只用了0.2.
相關文章
相關標籤/搜索