误区 #19:truncate表的操作不会被记录到日志
错误
在用户表中的操作都会被记录到日志。在sql server中唯一不会被记录到日志的操作是tempdb中的行版本控制。
truncate table语句会将整个表中的所有数据删除。但删除的方式并不是一行一行的删除,而是将组成表的数据页释放,将组成表的相关页释放的操作交给一个后台的线程进行队列处理的过程被称为deferred-drop。使用后台线程处理deferred-drop的好处是这个操作不会使得其所在的事务需要执行很长时间,因此也就不需要大量的锁。在sql server 2000sp3之前的版本(这个版本引入了deferred-drop)在truncate table的时候出现过多的锁耗尽内存的事是家常便饭。
下面是测试代码:
复制代码 代码如下:
create database truncatetest;
go
use truncatetest;
go
alter database truncatetest set recovery simple;
go
create table t1 (c1 int identity, c2 char (8000) default ‘a’);
create clustered index t1c1 on t1 (c1);
go
set nocount on;
go
insert into t1 default values;
go 1280
checkpoint;
go
上面的测试数据库恢复模式是简单,所以每个checkpoint都会截断日志(仅仅是为了简单,哈哈)。
一分钟后让我们来看看日志中有多少条记录。
复制代码 代码如下:
select count (*) from fn_dblog (null, null);
go
可以看到,现在的日志条目数字为2。
如果你得到的数字不是2,那么再做一次checkpoint直到数据是2为止。
现在已有的日志已经知道了,那么日志的增长就是由于后面的操作所导致。下面我们执行如下代码:
复制代码 代码如下:
truncate table t1;
go
select count (*) from fn_dblog (null, null);
go
可以看到现在已经有了541条日志记录。很明显truncate操作是需要记录到日志中的。但也可以看出truncate并不会逐行删除,因为这541条日志记录删除的是1280条数据。
执行下面语句来查看日志:
复制代码 代码如下:
select
[current lsn], [operation], [context],
[transaction id], [allocunitname], [transaction name]
from fn_dblog (null, null);
下面是结果:
图1.查看truncate后的日志(部分)
通过日志可以看出第一条显式开始truncate table事务,最后一条开始deferredalloc。正如你所见,truncate操作仅仅是释放了构成表的页和区。
下面这个代码可以查看日志具体所做操作的描述:
复制代码 代码如下:
select
[current lsn], [operation], [lock information], [description]
from fn_dblog (null, null);
go
结果如图2:
图2.日志操作描述(节选)
你可以看出为了快速恢复的目的而加的相关锁(你可以在我的博文:lock logging and fast recovery中了解更多)。
由上面日志看出,这个操作会对8个页加相关的锁,然后整个区一次性释放。释放过后会对相关的区加ix锁,也就是不能再被使用,当事务提交后才会进行deferred-drop,因此也就保证了truncate table操作可以回滚。
另外,如果表上存在非聚集索引.那么操作方式也是类似,都是交给一个后台线程然后释放表和索引的页。释放的最小单位就是每个分配单元。按照上面步骤你自己尝试一下就应该能明白我的意思了。
ps:还有一个关于truncate table操作不能回滚的误区,我在:search engine q&a #10: when are pages from a truncated table reused?这篇文章中进行了详细的解释。