实例对比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.
相关文章
相关标签/搜索