首先看一个例子:
--创建测试表
CREATETABLEtestfreespace
(column1INT
,column2CHAR(20)
,column3VARCHAR(8000))
--插入数据
DECLARE@countINT;
SET@count=0;
WHILE@count<3000
BEGIN
SELECT
@count=@count+1;
INSERTintotestfreespaceVALUES(@count,'testrow#'+CAST(@countASVARCHAR(10)),REPLICATE('TestData',3000));
END
3.--查看使用空间使用情况
SELECT
alloc_unit_type_desc,
page_count,
avg_page_space_used_in_percent,
record_count
FROMsys.dm_db_index_physical_stats(DB_ID('FNDBLogtest'),OBJECT_ID(N'Testfreespace'),NULL,NULL,'Detailed')
alloc_unit_type_descpage_countavg_page_space_used_in_percentrecord_count
----------------------------------------------------------------------------------------------------------------------------------
IN_ROW_DATA300099.27106498640973000
(1row(s)affected)
4.--删除整张表数据
deletefromtestfreespacewith(TABLOCK)
5.运行脚本3得到下面的
alloc_unit_type_descpage_countavg_page_space_used_in_percentrecord_count
----------------------------------------------------------------------------------------------------------------------------------
IN_ROW_DATA300000
可以看到虽然整张表的数据都被删除了但是表空间数据没有被释放。原因是SQLServerHEAP表空间释放需要两个条件:
·Adeletiononthistableoccurs.
·Atable-levellockisbeingheld.
释放空页时,数据库中的其他对象将无法重用关联的空间。
解决这个问题可以用下面的方法:
·在DELETE语句中指定TABLOCK提示。使用TABLOCK提示会导致删除操作获取表的共享锁,而不是行锁或页锁。这将允许释放页。
·如果要从表中删除所有行,请使用TRUNCATETABLE。
·删除行之前,请对堆创建聚集索引。删除行之后,可以删除聚集索引。与先前的方法相比,此方法非常耗时,并且使用更多的临时资源。
下面我用DELETEWITH(TABLOCK)删除表然后我们看一下结果:
--删除整张表数据
deletefromtestfreespacewith(TABLOCK)
alloc_unit_type_descpage_countavg_page_space_used_in_percentrecord_count
----------------------------------------------------------------------------------------------------------------------------------
IN_ROW_DATA000
另外建议表建立主键。(上面只针对HEAP表的情况)。