我有一个MS SQL Server 2008 Express系统,该系统包含一个我想“复制并重命名”(出于测试目的)的数据库,但是我不知道实现此目的的简单方法。
我注意到在R2版本的SQL Server中有一个复制数据库向导,但可惜我无法升级。
该数据库大约是演出。我试图将要复制的数据库备份还原到新数据库中,但是没有运气。
我有一个MS SQL Server 2008 Express系统,该系统包含一个我想“复制并重命名”(出于测试目的)的数据库,但是我不知道实现此目的的简单方法。
我注意到在R2版本的SQL Server中有一个复制数据库向导,但可惜我无法升级。
该数据库大约是演出。我试图将要复制的数据库备份还原到新数据库中,但是没有运气。
Answers:
安装Microsoft SQL Management Studio,您可以从Microsoft网站免费下载:
2008版
Microsoft SQL Management Studio 2008是带有高级服务的SQL Server 2008 Express的一部分
2012版
点击下载按钮并检查ENU\x64\SQLManagementStudio_x64_ENU.exe
2014版
单击下载按钮并检查MgmtStudio64BIT\SQLManagementStudio_x64_ENU.exe
打开Microsoft SQL Management Studio。
右键单击要克隆的数据库,单击Tasks
,然后单击Copy Database...
。按照向导完成操作。
您可以尝试分离数据库,在命令提示符下将文件复制到新名称,然后附加两个数据库。
在SQL中:
USE master;
GO
EXEC sp_detach_db
@dbname = N'OriginalDB';
GO
在命令提示符下(为简化此示例,我简化了文件路径):
copy c:\OriginalDB.mdf c:\NewDB.mdf
copy c:\OriginalDB.ldf c:\NewDB.ldf
再次在SQL中:
USE master;
GO
CREATE DATABASE OriginalDB
ON (FILENAME = 'C:\OriginalDB.mdf'),
(FILENAME = 'C:\OriginalDB.ldf')
FOR ATTACH;
GO
CREATE DATABASE NewDB
ON (FILENAME = 'C:\NewDB.mdf'),
(FILENAME = 'C:\NewDB.ldf')
FOR ATTACH;
GO
select * from OriginalDB.sys.sysfiles
查找数据库文件的位置。
Permission denied
在.mdf
文件中说。我现在不需要它,我只需要备份原始数据库,因此以后可以覆盖原始数据库,我很好奇为什么会出现这样的错误。
exec sp_helpdb @dbname='TEMPDB';
事实证明,我试图从备份中错误地还原。
最初,我创建了一个新数据库,然后尝试在此处还原备份。我应该做的,最后起作用的是,弹出还原对话框,并在目标字段中键入新数据库的名称。
因此,简而言之,从备份还原就可以了。
谢谢大家的反馈和建议
这是我使用的脚本。有点棘手,但有效。在SQL Server 2012上测试。
DECLARE @backupPath nvarchar(400);
DECLARE @sourceDb nvarchar(50);
DECLARE @sourceDb_log nvarchar(50);
DECLARE @destDb nvarchar(50);
DECLARE @destMdf nvarchar(100);
DECLARE @destLdf nvarchar(100);
DECLARE @sqlServerDbFolder nvarchar(100);
SET @sourceDb = 'db1'
SET @sourceDb_log = @sourceDb + '_log'
SET @backupPath = 'E:\tmp\' + sourceDb + '.bak' --ATTENTION: file must already exist and SQL Server must have access to it
SET @sqlServerDbFolder = 'E:\DB SQL\MSSQL11.MSSQLSERVER\MSSQL\DATA\'
SET @destDb = 'db2'
SET @destMdf = @sqlServerDbFolder + @destDb + '.mdf'
SET @destLdf = @sqlServerDbFolder + @destDb + '_log' + '.ldf'
BACKUP DATABASE @sourceDb TO DISK = @backupPath
RESTORE DATABASE @destDb FROM DISK = @backupPath
WITH REPLACE,
MOVE @sourceDb TO @destMdf,
MOVE @sourceDb_log TO @destLdf
SET @sourceDb_log = (SELECT files.name FROM sys.databases dbs INNER JOIN sys.master_files files ON dbs.database_id=files.database_id WHERE dbs.name=@sourceDb AND files.type=1)
一个@sourceDb_data的单独变量,并带有类似的查询(替换为files.type=0
)。HTH!
此处提到的解决方案都不适用于我-我正在使用SQL Server Management Studio 2014。
取而代之的是,我必须取消选中“选项”屏幕中的“还原前先进行尾日志备份”复选框:在我的版本中,默认情况下已选中该复选框并阻止还原操作完成。取消选中后,还原操作将继续进行,不会出现问题。
使用MS SQL Server 2012,您需要执行3个基本步骤:
首先,生成.sql
仅包含源数据库结构的文件
.sql
本地文件第二,将源数据库替换为.sql
文件中的目标数据库
最后,填充数据
Data Source=Mehdi\SQLEXPRESS;Initial Catalog=db_test;User ID=sa;Password=sqlrpwrd15
大功告成
如果数据库不是很大,则可以查看SQL Server Management Studio Express中的“脚本数据库”命令,该命令位于资源管理器中数据库项本身旁边的上下文菜单中。
您可以选择全部编写脚本;当然,您需要对象和数据。然后,您将整个脚本保存到单个文件中。然后,您可以使用该文件重新创建数据库。只需确保将USE
顶部的命令设置为正确的数据库即可。
基于此注释的解决方案:https : //stackoverflow.com/a/22409447/2399045。只需设置设置:数据库名称,临时文件夹,数据库文件文件夹。运行后,您将获得名称为“ sourceDBName_yyyy-mm-dd”格式的数据库副本。
-- Settings --
-- New DB name will have name = sourceDB_yyyy-mm-dd
declare @sourceDbName nvarchar(50) = 'MyDbName';
declare @tmpFolder nvarchar(50) = 'C:\Temp\'
declare @sqlServerDbFolder nvarchar(100) = 'C:\Databases\'
-- Execution --
declare @sourceDbFile nvarchar(50);
declare @sourceDbFileLog nvarchar(50);
declare @destinationDbName nvarchar(50) = @sourceDbName + '_' + (select convert(varchar(10),getdate(), 121))
declare @backupPath nvarchar(400) = @tmpFolder + @destinationDbName + '.bak'
declare @destMdf nvarchar(100) = @sqlServerDbFolder + @destinationDbName + '.mdf'
declare @destLdf nvarchar(100) = @sqlServerDbFolder + @destinationDbName + '_log' + '.ldf'
SET @sourceDbFile = (SELECT top 1 files.name
FROM sys.databases dbs
INNER JOIN sys.master_files files
ON dbs.database_id = files.database_id
WHERE dbs.name = @sourceDbName
AND files.[type] = 0)
SET @sourceDbFileLog = (SELECT top 1 files.name
FROM sys.databases dbs
INNER JOIN sys.master_files files
ON dbs.database_id = files.database_id
WHERE dbs.name = @sourceDbName
AND files.[type] = 1)
BACKUP DATABASE @sourceDbName TO DISK = @backupPath
RESTORE DATABASE @destinationDbName FROM DISK = @backupPath
WITH REPLACE,
MOVE @sourceDbFile TO @destMdf,
MOVE @sourceDbFileLog TO @destLdf
基于Joe答案的脚本(分离,复制文件,同时附加两者)。
这不是必需的,但执行时可能会拒绝访问。
EXEC sp_configure 'show advanced options', 1
GO
RECONFIGURE
GO
EXEC sp_configure 'xp_cmdshell', 1
GO
RECONFIGURE
GO
@dbName
和@copyDBName
之前输入数据库名称。USE master;
GO
DECLARE @dbName NVARCHAR(255) = 'Products'
DECLARE @copyDBName NVARCHAR(255) = 'Products_branch'
-- get DB files
CREATE TABLE ##DBFileNames([FileName] NVARCHAR(255))
EXEC('
INSERT INTO ##DBFileNames([FileName])
SELECT [filename] FROM ' + @dbName + '.sys.sysfiles')
-- drop connections
EXEC('ALTER DATABASE ' + @dbName + ' SET OFFLINE WITH ROLLBACK IMMEDIATE')
EXEC('ALTER DATABASE ' + @dbName + ' SET SINGLE_USER')
-- detach
EXEC('EXEC sp_detach_db @dbname = ''' + @dbName + '''')
-- copy files
DECLARE @filename NVARCHAR(255), @path NVARCHAR(255), @ext NVARCHAR(255), @copyFileName NVARCHAR(255), @command NVARCHAR(MAX) = ''
DECLARE
@oldAttachCommand NVARCHAR(MAX) =
'CREATE DATABASE ' + @dbName + ' ON ',
@newAttachCommand NVARCHAR(MAX) =
'CREATE DATABASE ' + @copyDBName + ' ON '
DECLARE curs CURSOR FOR
SELECT [filename] FROM ##DBFileNames
OPEN curs
FETCH NEXT FROM curs INTO @filename
WHILE @@FETCH_STATUS = 0
BEGIN
SET @path = REVERSE(RIGHT(REVERSE(@filename),(LEN(@filename)-CHARINDEX('\', REVERSE(@filename),1))+1))
SET @ext = RIGHT(@filename,4)
SET @copyFileName = @path + @copyDBName + @ext
SET @command = 'EXEC master..xp_cmdshell ''COPY "' + @filename + '" "' + @copyFileName + '"'''
PRINT @command
EXEC(@command);
SET @oldAttachCommand = @oldAttachCommand + '(FILENAME = "' + @filename + '"),'
SET @newAttachCommand = @newAttachCommand + '(FILENAME = "' + @copyFileName + '"),'
FETCH NEXT FROM curs INTO @filename
END
CLOSE curs
DEALLOCATE curs
-- attach
SET @oldAttachCommand = LEFT(@oldAttachCommand, LEN(@oldAttachCommand) - 1) + ' FOR ATTACH'
SET @newAttachCommand = LEFT(@newAttachCommand, LEN(@newAttachCommand) - 1) + ' FOR ATTACH'
-- attach old db
PRINT @oldAttachCommand
EXEC(@oldAttachCommand)
-- attach copy db
PRINT @newAttachCommand
EXEC(@newAttachCommand)
DROP TABLE ##DBFileNames