在公司的內網有臺數據庫的測試服務器,這臺服務器是提供給開發人員使用的,在上面有不少的數據庫,有些是臨時系統用到的數據庫,這些數據庫有一個共同點:數據庫表結構比較重要,數據庫只有一些測試數據,也就是說這些數據庫都很小,而整臺服務器的數據庫又很是多;html
如今有這樣一個需求,但願間隔一段時間就備份全部數據庫,因此這裏寫了這篇文章,這也是另一篇文章SQL Server 批量主分區備份(One Job)的基礎;sql
下面是實現批量備份數據庫的3種方式,你們能夠細細體會其中的差異:數據庫
1) 實現方式1:使用遊標編程
2) 實現方式2:使用拼湊SQL的方式服務器
3) 實現方式3:使用存儲過程sp_MSforeachdb_Filter(以sp_MSforeachdb爲基礎)函數
執行下面的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的目錄下生成相似下圖的備份文件:加密
(Figure1:數據庫備份文件)spa
--使用拼湊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所示:
(Figure2:生成的T-SQL腳本)
(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的區別有如下兩點:
(Figure4:添加內容1)
(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的效果以下圖所示:
(Figure6:錯誤信息)
若是沒有設置SET QUOTED_IDENTIFIER 這個選項爲 OFF ,那麼在調用存儲過程sp_MSforeachdb_Filter的時候會出現下圖所示的錯誤信息:
(Figure7:錯誤信息)
若是想查看存儲過程sp_MSforeachdb的詳細代碼,能夠在經過訪問路徑:數據庫-可編程性-存儲過程-系統存儲過程-sp_MSforeachdb找到,或者經過下面的腳本查看:
--顯示規則、默認值、未加密的存儲過程、用戶定義函數、觸發器或視圖的文本 EXEC sp_helptext N'sp_MSforeachdb';
更多批量備份數據庫的文章能夠參考:
SQL Server批量主分區備份(Multiple Jobs)