gpt4 book ai didi

sql-server - 如何使用 sqlcmd 在批处理文件中显示进度?

转载 作者:行者123 更新时间:2023-12-02 21:12:22 24 4
gpt4 key购买 nike

我正在使用 sqlcmd ( sqlcmd reference ) 在 dos 批处理文件(不是 powershell)中备份大型数据库。大约需要30分钟。

sqlcmd -S 127.0.0.1 -d DbNameHere -E -Q "BACKUP DATABASE [DbNameHere] TO DISK = N'c:\Temp\MyBackup.bak' WITH COPY_ONLY, NOFORMAT, INIT, NAME = N'My Full Database Backup', SKIP, NOREWIND, NOUNLOAD, STATS = 10"

如果您在 SQL Management Studio 中运行 BACKUP 命令,您会得到如下输出:

10 percent processed.
20 percent processed.
...

在 DOS 批处理中,最好的情况是,当备份完成时,我会立即将所有 10,20,30..100 全部显示在屏幕上。

我已经尝试使用这些参数,但我仍然没有在屏幕上获得所需的进度更新:

-m-1
-V 1
-r1

这些进度消息被缓冲,这可能是问题的一部分。例如,此处讨论:How do I flush the PRINT buffer in TSQL?但我有一个长时间运行的命令,而不是多个命令。

您可以在其他地方运行单独的 SQL 语句,它会告诉您进度甚至预计完成时间:

SELECT session_id as SPID, command, a.text AS Query, start_time, percent_complete, dateadd(second,estimated_completion_time/1000, getdate()) as estimated_completion_time 
FROM sys.dm_exec_requests r CROSS APPLY sys.dm_exec_sql_text(r.sql_handle) a
WHERE r.command in ('BACKUP DATABASE','RESTORE DATABASE')

但是要使用它,我必须创建第二个批处理文件(使用 sqlcmd 执行此语句),在运行 sqlcmd 备份之前在新窗口中打开它,并在 1 分钟计时器上循环运行它也许吧,并在备份完成后结束它。所有这些都是批量的。理想情况下,我宁愿将其全部保存在一个批处理文件中!输出标准进度消息会简单得多!

有什么想法吗?

最佳答案

我想出了一种解决方案。缺点是它涉及第二个批处理文件。它基本上启动一个带有给定名称的进度窗口。该进度窗口会定期刷新。主进程完成后,它会使用给定的名称调用taskkill。进度批处理依赖于超时。两者都在我的 Windows 8 工作站以及 2008 R2 和 2012 服务器上。

主备份批处理:

@echo off
rem lots of other lines in my main batch file here

start "BACKUP_PROGRESS" /belownormal cmd /d /c "backup_progress2.cmd"

sqlcmd -S 127.0.0.1 -d DBNameHere-E -Q "BACKUP DATABASE [DBNameHere] TO DISK = N'c:\Temp\DBNameHere.bak' WITH COPY_ONLY, NOFORMAT, INIT, NAME = N'DBNameHere-Full Database Backup', SKIP, NOREWIND, NOUNLOAD, STATS = 10"

taskkill /FI "WINDOWTITLE eq BACKUP_PROGRESS" /F > nul

rem my main batch file continues here with lots of other lines

进度批处理文件(在主批处理中称为 backup_progress2.cmd)。 setlocal enableextensionsenabledelayedexpansion 只是为了让我可以使用 !time! 显示以秒为单位的时间。

@echo off
rem http://stackoverflow.com/questions/21434982/windows-batch-scripting-catch-user-reaction-to-timeout-command
SET timeout=60

:loop
cls
setlocal enableextensions enabledelayedexpansion
echo.
echo Time now: !time!
echo.
endlocal
sqlcmd -S 127.0.0.1 -d DBNameHere -E -Q "SET NOCOUNT ON;SELECT start_time,cast(percent_complete as int) as progress,dateadd(second,estimated_completion_time/1000, getdate()) as estimated_completion_time, cast(estimated_completion_time/1000/60 as int) as minutes_left FROM sys.dm_exec_requests r WHERE r.command='BACKUP DATABASE'"
echo.
echo Refreshing every %timeout% seconds.
timeout %timeout% > nul
goto loop

关于sql-server - 如何使用 sqlcmd 在批处理文件中显示进度?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31265434/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com