SQL Server误区30日谈 第19天 Truncate表的操作不会被记录到日志
韩国旅游多少钱,圣力皇决txt下载,莫菁门
误区 #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.日志操作描述(节选)
你可以看出为了快速恢复的目的而加的相关锁(你可以在我的博文:中了解更多)。
由上面日志看出,这个操作会对8个页加相关的锁,然后整个区一次性释放。释放过后会对相关的区加ix锁,也就是不能再被使用,当事务提交后才会进行deferred-drop,因此也就保证了truncate table操作可以回滚。
另外,如果表上存在非聚集索引.那么操作方式也是类似,都是交给一个后台线程然后释放表和索引的页。释放的最小单位就是每个分配单元。按照上面步骤你自己尝试一下就应该能明白我的意思了。
ps:还有一个关于truncate table操作不能回滚的误区,我在:这篇文章中进行了详细的解释。
如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!!
点击进行留言回复
相关文章:
-
-
-
sql某个日期是当年的第几周
/* *周一作为一周的开始 *当年的1月1号所在的周算作第一周 */ CREATE function GetWeekIndexFirstDate ( @...
[阅读全文]
-
-
数据库SQL---范式
1、数据冗余导致的问题:冗余存储、更新异常、插入异常、删除异常。 2、函数依赖:一种完整性约束。 在关系模式r(R)中,α属于R,β属于R。 1)α函数...
[阅读全文]
-
-
数据库SQL---查询
1、查询所有列 select *from emp;--*表示所有的,from emp表示从emp表中查询。 2、查询指定列 select empno,e...
[阅读全文]
-
-
-
-
网友评论