我有一个数据库的测试环境,我想在测试周期开始时用新数据重新加载该数据库。我对重建整个数据库不感兴趣——只是简单地“重新设置”数据。

使用TSQL从所有表中删除所有数据的最佳方法是什么?是否有可以使用的系统存储过程、视图等?我不想为每个表手动创建和维护截断表语句-我更希望它是动态的。


当前回答

最简单的方法就是

打开SQL Management Studio 导航到数据库 右键单击并选择任务->生成脚本(pic 1) 在“选择对象”界面,选择“选择特定对象”选项并勾选“表格”(图2) 在下一个屏幕上,选择“高级”,然后将“Script DROP and CREATE”选项更改为“Script DROP and CREATE”(图3) 选择将脚本保存到新的编辑器窗口或文件中,并根据需要运行。

这将为您提供一个删除和重新创建所有表的脚本,而无需担心调试或是否包含了所有内容。虽然这不仅仅执行截断,但结果是相同的。请记住,自动递增的主键将从0开始,而截断的表将记住最后分配的值。如果在PreProd或Production环境中无法访问Management studio,也可以从代码中执行此操作。

1.

2.

3.

其他回答

如果你想在一个特定的表(即静态查找表)中保留数据,同时删除/截断同一数据库中其他表中的数据,那么你需要一个循环,其中包含异常。这就是我在无意中发现这个问题时所寻找的。

sp_MSForEachTable对我来说似乎有bug(即与IF语句不一致的行为),这可能是为什么它没有被MS记录的原因。

declare @LastObjectID int = 0
declare @TableName nvarchar(100) = ''
set @LastObjectID = (select top 1 [object_id] from sys.tables where [object_id] > @LastObjectID order by [object_id])
while(@LastObjectID is not null)
begin
    set @TableName = (select top 1 [name] from sys.tables where [object_id] = @LastObjectID)

    if(@TableName not in ('Profiles', 'ClientDetails', 'Addresses', 'AgentDetails', 'ChainCodes', 'VendorDetails'))
    begin
        exec('truncate table [' + @TableName + ']')
    end 

    set @LastObjectID = (select top 1 [object_id] from sys.tables where [object_id] > @LastObjectID order by [object_id])
end

制作一个空的“模板”数据库,进行完全备份。当您需要刷新时,只需使用WITH REPLACE恢复。快速,简单,防弹。如果这里或那里的几个表需要一些基本数据(例如。配置信息,或让你的应用程序运行的基本信息),它也会处理这些。

这是一种方法……可能还有其他10种更好/更有效的方法,但这听起来似乎很少有人这样做,所以下面是……

从sysobjects中获取一个表列表,然后用游标遍历这些表,为每次迭代调用sp_execsql('truncate table ' + @table_name)。

对于SQL 2005,

EXEC sp_MSForEachTable 'TRUNCATE TABLE ?'

2000年和2005/2008年的链接更多。

只有当你的表之间没有任何外键关系时,才能截断所有的表,因为SQL Server不允许你用外键截断一个表。

另一种方法是先确定有外键的表并删除它们,然后再截断没有外键的表。

详见http://www.sqlteam.com/forums/topic.asp?TOPIC_ID=65341和http://www.sqlteam.com/forums/topic.asp?TOPIC_ID=72957。