SQL Server 批量完整備份

一.本文所涉及的內容(Contents)

  1. 本文所涉及的內容(Contents)
  2. 背景(Contexts)
  3. 實現代碼(SQL Codes)
    1. 實現方式一(One)
    2. 實現方式二(Two)
    3. 實現方式三(Three)
  4. 參考文獻(References)

二.背景(Contexts)

  在公司的內網有臺數據庫的測試服務器,這臺服務器是提供給開發人員使用的,在上面有不少的數據庫,有些是臨時系統用到的數據庫,這些數據庫有一個共同點:數據庫表結構比較重要,數據庫只有一些測試數據,也就是說這些數據庫都很小,而整臺服務器的數據庫又很是多;html

  如今有這樣一個需求,但願間隔一段時間就備份全部數據庫,因此這裏寫了這篇文章,這也是另一篇文章SQL Server 批量主分區備份(One Job)的基礎;sql

三.實現代碼(SQL Codes)

下面是實現批量備份數據庫的3種方式,你們能夠細細體會其中的差異:數據庫

1) 實現方式1:使用遊標編程

2) 實現方式2:使用拼湊SQL的方式服務器

3) 實現方式3:使用存儲過程sp_MSforeachdb_Filter(以sp_MSforeachdb爲基礎)函數

 

(一)實現方式1:使用遊標post

執行下面的SQL腳本就能夠備份當前數據庫實例的全部數據庫(除了系統數據庫);測試

-- =============================================
-- Author:      <聽風吹雨>
-- Blog:        <http://gaizai.cnblogs.com/>
-- Create date: <2011/12/03>
-- Description: <批量備份數據庫>
-- =============================================
DECLARE
      @FileName VARCHAR(200),
      @CurrentTime VARCHAR(50),
      @DBName VARCHAR(100),
      @SQL VARCHAR(1000)

SET @CurrentTime = CONVERT(CHAR(8),GETDATE(),112) + CAST(DATEPART(hh, GETDATE()) AS VARCHAR) + CAST(DATEPART(mi, GETDATE()) AS VARCHAR)

DECLARE CurDBName CURSOR FOR 
    SELECT NAME FROM Master..SysDatabases where dbid>4

OPEN CurDBName
FETCH NEXT FROM CurDBName INTO @DBName
WHILE @@FETCH_STATUS = 0
BEGIN
    --Execute Backup
    SET @FileName = 'E:\DBBackup\' + @DBName + '_' + @CurrentTime
    SET @SQL = 'BACKUP DATABASE ['+ @DBName +'] TO DISK = ''' + @FileName + '.bak' +
     ''' WITH NOINIT, NOUNLOAD, NAME = N''' + @DBName + '_backup'', NOSKIP, STATS = 10, NOFORMAT'
    EXEC(@SQL)

    --Get Next DataBase
    FETCH NEXT FROM CurDBName INTO @DBName
END

CLOSE CurDBName
DEALLOCATE CurDBName

執行完上面的SQL腳本,會在E:\DBBackup的目錄下生成相似下圖的備份文件:加密

clip_image002

(Figure1:數據庫備份文件)spa

 

(二)實現方式2:使用拼湊SQL的方式

--使用拼湊SQL的方式
DECLARE @SQL VARCHAR(MAX)

SELECT @SQL = COALESCE(@SQL,'') + '
BACKUP DATABASE '+ QUOTENAME(name,'[]') 
+ ' TO DISK = ''E:\DBBackup\'+ name + '_' + CONVERT(CHAR(8),GETDATE(),112) + CAST(DATEPART(hh, GETDATE()) AS VARCHAR) + CAST(DATEPART(mi, GETDATE()) AS VARCHAR) + '.bak' 
+ ''' WITH NOINIT, NOUNLOAD, NAME = N''' + name + '_backup'', NOSKIP, STATS = 10, NOFORMAT'
FROM sys.databases WHERE database_id >4 AND name like '%%' AND state =0

PRINT(@SQL)
EXECUTE(@SQL)

生成的腳本如Figure2所示,若是想腳本更加美觀,能夠加上GO語句,如Figure3所示:

clip_image004

(Figure2:生成的T-SQL腳本)

clip_image006

(Figure3:生成的T-SQL腳本)

 

(三)實現方式3:使用存儲過程sp_MSforeachdb_Filter(以sp_MSforeachdb爲基礎)

經過查看系統存儲過程sp_MSforeachdb的T-SQL源代碼能夠發現是沒有提供@whereand參數能夠過濾數據庫的,參考系統存儲過程sp_MSforeachtable後,在sp_MSforeachdb的基礎上建立帶@whereand參數的存儲過程sp_MSforeachdb_Filter,這樣你就可讓SQL在指定的數據庫上執行;

-- =============================================
-- Author:      <聽風吹雨>
-- Blog:        <http://gaizai.cnblogs.com/>
-- Create date: <2013.05.06>
-- Description: <擴展sp_MSforeachdb,增長@whereand參數>
-- =============================================
USE [master]
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER OFF
GO
 
create proc [dbo].[sp_MSforeachdb_Filter]
    @command1 nvarchar(2000), @replacechar nchar(1) = N'?', @command2 nvarchar(2000) = null, @command3 nvarchar(2000) = null,
    @whereand nvarchar(2000) = null,@precommand nvarchar(2000) = null, @postcommand nvarchar(2000) = null
as
    set deadlock_priority low
    
    /* This proc returns one or more rows for each accessible db, with each db defaulting to its own result set */
    /* @precommand and @postcommand may be used to force a single result set via a temp table. */
 
    /* Preprocessor won't replace within quotes so have to use str(). */
    declare @inaccessible nvarchar(12), @invalidlogin nvarchar(12), @dbinaccessible nvarchar(12)
    select @inaccessible = ltrim(str(convert(int, 0x03e0), 11))
    select @invalidlogin = ltrim(str(convert(int, 0x40000000), 11))
    select @dbinaccessible = N'0x80000000'        /* SQLDMODbUserProf_InaccessibleDb; the negative number doesn't work in convert() */
 
    if (@precommand is not null)
        exec(@precommand)
 
    declare @origdb nvarchar(128)
    select @origdb = db_name()
 
    /* If it's a single user db and there's an entry for it in sysprocesses who isn't us, we can't use it. */
   /* Create the select */
    exec(N'declare hCForEachDatabase cursor global for select name from master.dbo.sysdatabases d ' +
            N' where (d.status & ' + @inaccessible + N' = 0)' +
            N' and (DATABASEPROPERTY(d.name, ''issingleuser'') = 0 and (has_dbaccess(d.name) = 1))' + @whereand)
 
    declare @retval int
    select @retval = @@error
    if (@retval = 0)
        exec @retval = sys.sp_MSforeach_worker @command1, @replacechar, @command2, @command3, 1
 
    if (@retval = 0 and @postcommand is not null)
        exec(@postcommand)
 
   declare @tempdb nvarchar(258)
   SELECT @tempdb = REPLACE(@origdb, N']', N']]')
   exec (N'use ' + N'[' + @tempdb + N']')
 
    return @retval

上面的存儲過程sp_MSforeachdb_Filter與sp_MSforeachdb的區別有如下兩點:

clip_image008

(Figure4:添加內容1)

clip_image010

(Figure5:添加內容2)

並且須要注意在建立存儲過程的時候須要設置SET QUOTED_IDENTIFIER OFF,當 SET QUOTED_IDENTIFIER 爲 ON 時,標識符能夠由雙引號分隔,而文字必須由單引號分隔;當 SET QUOTED_IDENTIFIER 爲 OFF 時,標識符不可加引號,且必須符合全部 Transact-SQL 標識符規則。具體能夠參考:SET QUOTED_IDENTIFIER (Transact-SQL)

調用sp_MSforeachdb_Filter實現批量備份數據庫的T-SQL以下所示:

--使用更新的存儲過程sp_MSforeachdb_Filter(以sp_MSforeachdb爲基礎)
USE [master]
GO

DECLARE @SQL NVARCHAR(MAX)
SELECT @SQL = COALESCE(@SQL,'') + '
BACKUP DATABASE [?] 
TO DISK = ''E:\DBBackup\?_' + CONVERT(CHAR(8),GETDATE(),112) + CAST(DATEPART(hh, GETDATE()) AS VARCHAR) + CAST(DATEPART(mi, GETDATE()) AS VARCHAR) + '.bak'' 
WITH NOINIT, NOUNLOAD, NAME = N''?_backup'', NOSKIP, STATS = 10, NOFORMAT'

PRINT @SQL

--過濾數據庫
EXEC [sp_MSforeachdb_Filter] @command1=@SQL,
@whereand=" and [name] not in('tempdb','master','model','msdb') "

執行上面的存儲過程就能夠備份全部數據庫(系統數據庫除外,想要過濾數據庫能夠填寫@whereand參數的條件),執行上面SQL的效果以下圖所示:

clip_image011

(Figure6:錯誤信息)

若是沒有設置SET QUOTED_IDENTIFIER 這個選項爲 OFF ,那麼在調用存儲過程sp_MSforeachdb_Filter的時候會出現下圖所示的錯誤信息:

clip_image013

(Figure7:錯誤信息)

若是想查看存儲過程sp_MSforeachdb的詳細代碼,能夠在經過訪問路徑:數據庫-可編程性-存儲過程-系統存儲過程-sp_MSforeachdb找到,或者經過下面的腳本查看:

--顯示規則、默認值、未加密的存儲過程、用戶定義函數、觸發器或視圖的文本
EXEC sp_helptext N'sp_MSforeachdb';

更多批量備份數據庫的文章能夠參考:

SQL Server 批量主分區備份(One Job)

SQL Server批量主分區備份(Multiple Jobs)

四.參考文獻(References)

SET QUOTED_IDENTIFIER (Transact-SQL)(英文)

SET QUOTED_IDENTIFIER (Transact-SQL)(中文)

相關文章
相關標籤/搜索