如何备份和恢复表

问题描述 投票:0回答:4

在测试过程中,我想在运行测试之前复制同一数据库中的几个表。测试完成后,我想用副本恢复原始表。

最好的方法是什么?
我还想确保所有索引和约束都已恢复。

DECLARE @Tablename NVARCHAR(500)
DECLARE @BuildStr NVARCHAR(500)
DECLARE @SQL NVARCHAR(500)
SELECT @Tablename = 'my_Users'
SELECT @BuildStr = CONVERT(NVARCHAR(16),GETDATE(),120)
SELECT @BuildStr = REPLACE(REPLACE(REPLACE(REPLACE(@BuildStr,' 
',''),':',''),'-',''),' ','')
SET @SQL = 'select * into '+@Tablename+'_'+@BuildStr+' from '+@Tablename
SELECT @SQL

EXEC (@SQL) -- Execute SQl statement

如果我使用上述方法进行复制,如何恢复。

SQL2005

sql sql-server t-sql backup
4个回答
2
投票

有很多方法可以做到这一点,但到目前为止,最简单的方法是简单地备份数据库,使用它,然后在完成后从备份恢复。 (这里有说明

备份表当然是可行的,但这不是最简单的方法,一旦你开始使用多个表,它就会变得更加困难。因此,我不是解决恢复单个表的特定示例,而是提供关于更好地管理测试数据的一般建议。

最安全的方法是不恢复原始文件,而是甚至不碰原始文件。对其进行备份,然后将其恢复到新的测试服务器。 (此处的说明)最佳实践表明,无论如何,您都不应该在实时数据库上进行测试或开发工作。这也很简单,而且安全。


2
投票

我认为我最近使用的脚本可能对某人有用。

要备份表,您可以使用下一个查询:

DECLARE @tableName nvarchar(max), @tableName_bck nvarchar(max)
SET @tableName = 'SomeTable';
SET @tableName_bck = 'SomeTable_bck';


-- Backup
DECLARE @insertCommand nvarchar(max)
--SELECT INTO SomeTable_bck FROM SomeTable
SET @insertCommand = 'SELECT * INTO ' + @tableName_bck + ' FROM ' + @tableName
PRINT @insertCommand
EXEC sp_executesql @insertCommand

对于恢复,因为表通常可以有 IDENTITY 字段,所以需要 SET IDENTITY_INSERT ON 并且在插入记录时需要提供列列表。这就是为什么脚本有点复杂:

DECLARE @tableName nvarchar(max), @tableName_bck nvarchar(max)
SET @tableName = 'SomeTable';
SET @tableName_bck = 'SomeTable_bck';

-- Restore
DECLARE @columnList nvarchar(max)
DECLARE @insertCommand nvarchar(max)

SELECT
    @columnList = SUBSTRING(
        (
            SELECT ', ' + column_name AS [text()]
            From INFORMATION_SCHEMA.COLUMNS
            WHERE table_name = @tableName
            ORDER BY table_name
            For XML PATH ('')
        ), 2, 1000);

--INSERT INTO SomeTable(Column1, Column2) SELECT Column1, Column2 FROM SomeTable_bck        
SELECT @insertCommand = 'INSERT INTO ' + @tableName + '(' + @columnList + ') SELECT ' + @columnList + ' FROM ' + @tableName_bck

IF EXISTS (
    SELECT column_name, table_name
    FROM INFORMATION_SCHEMA.COLUMNS
       WHERE table_schema = 'dbo' AND table_name = @tableName
       AND COLUMNPROPERTY(object_id(table_name), column_name, 'IsIdentity') = 1
    )
BEGIN
    SET @insertCommand = 
          'SET IDENTITY_INSERT ' + @tableName + ' ON;' 
        + 'TRUNCATE TABLE ' + @tableName + ';'
        + @insertCommand + ';'
        + 'SET IDENTITY_INSERT ' + @tableName + ' OFF;' 
    /*  
    SET IDENTITY_INSERT SomeTable ON
    TRUNCATE TABLE SomeTable
    INSERT INTO SomeTable(Column1, Column2) SELECT Column1, Column2 FROM SomeTable_bck    
    SET IDENTITY_INSERT SomeTable OFF
    */
END
ELSE
BEGIN
    SET @insertCommand = 
        'TRUNCATE TABLE ' + @tableName + ';'
        + @insertCommand
    /*
    TRUNCATE TABLE SomeTable
    INSERT INTO SomeTable(Column1, Column2) SELECT Column1, Column2 FROM SomeTable_bck     
    */
END 

PRINT @insertCommand
EXEC sp_executesql @insertCommand

很容易看出,您可以根据自己的喜好指定@tableName和@tableName_bck。例如,这可以在存储过程中,因此脚本是可重用的。


2
投票

首先,将原表复制到新表中:

CREATE TABLE CopiedTable AS SELECT * FROM OriginalTable;

每当您想要恢复时,截断原始数据并将值从副本插入回:

TRUNCATE TABLE OriginalTable
INSERT INTO OriginalTable SELECT * FROM CopiedTable

根据您使用的数据库,有更快的替代方案。

另外,要小心。正如其他人指出的,最安全的方法是备份孔数据库。


1
投票

您是否考虑过使用 SQL Server 单元测试框架,例如开源 tSQLt 框架?

参见 http://tsqlt.org/

tSQLt 测试在事务中运行,因此您在测试中所做的任何操作都将回滚。

它有一个“faketable”的概念,它是原始表减去约束的副本(如果这些约束妨碍了您的测试设置)。

© www.soinside.com 2019 - 2024. All rights reserved.