如何避免脚本的多个实例,同时允许在不更改标题的情况下重新启动它



我有这个脚本,它应该只在单个实例中运行。 我确保这种情况不会发生的方法是

tasklist /fi "imagename eq cmd.exe" /v | find "lootbot" && echo already running && pause && exit /b
title lootbot

稍后执行一个优雅的退出,以便可以重新启动脚本...

:action8
cd /d "%~dp0"
title command prompt
echo type launch to return to lootbot
exit /b

问题是,每当脚本因错误而崩溃时,如果不先手动将标题重新设置为其他内容,它就不允许我启动它。

当然,这不是一个大问题,但我正在构建它供其他人使用,所以我在这里寻找一个易于使用的解决方案。我在想是否有办法确定当前命令提示符是否标题为lootbot那么可以自动避免此问题。这是整个脚本以防万一。

@echo off
tasklist /fi "imagename eq cmd.exe" /v | find "lootbot" && echo lootbot already running, press a key to exit && pause && exit /b
title lootbot
:: enable save files location (persistant memory)
set "saves=%appdata%lootbot"
if not exist "%saves%" mkdir "%saves%"
:: create/load saves, echo first
if exist "%saves%echoeslaunch" echo on
if not exist "%saves%loopdelay" echo 300>"%saves%loopdelay"
if not exist "%saves%promptdelay" echo 60>"%saves%promptdelay"
set /p loopdelay=<"%saves%loopdelay" & set /p promptdelay=<"%saves%promptdelay"
:: advanced prep
chcp 65001 >nul
set "path=%path%;%~dp0;c:program filesfilebot"
wmic process where name="cmd.exe" call setpriority "idle" >nul
:: additional scripts (all-in-one launcher)
start vpn.cmd
:: add "going legit" to vpn or somewhere (meaning close torrents and vpn, reactivate vpn if torrents appear, or close torrents)
:: add skippable delay here to auto-avoid possible mishandling ones being checked in qbittorrent
cls
:menu
@colorx -c 08
@if exist "%saves%echoeslaunch" echo on
@cd /d "%~dp0"
@set "msg=lootbot ready (%time:~0,2%:%time:~3,2%) wait %loopdelay%s or enter command"
@set "msg=%msg:( =(%" & rem this removes the blank space from early hours
@choice /t %loopdelay% /c rcplsjdq /n /d r /m "%msg% [R]un [C]lear [P]ause [L]ist [S]etup [J]obs [D]onate [Q]uit: "
:: add selective choices here, based on what scripts are available. might need those scripts to be names or placed different to autodetect
@goto action%errorlevel%
@goto menu
:action1
:routine
:: run now/default infinite loop
call report %~n0 import
call report %~n0 direct
:: idea: add small amount of random files to scan?
goto menu
:action2
:clear
cls
goto menu
:action3
pause
goto menu
:action4
:playlist
call report %~n0 player
goto menu
:action5
:settings
call report %~n0 option
goto menu
:action6
:caring
call report %~n0 caring
goto menu
:action7
:donate
call report %~n0 donate
goto menu
:action8
:exit
:: quit back to command prompt
cd /d "%~dp0"
title command prompt
echo use command launch to return to lootbot, bye!
exit /b

通过战略性地使用 CALL 重定向来建立锁定文件,这很容易完成。只有一个实例可以对锁定文件具有写入访问权限。第二个实例的尝试将失败。第一个实例将在退出后立即释放锁,无论它如何终止。

我使用多个重定向阶段来隐藏不需要的错误消息,但主例程具有正常的标准输出和标准输出。我添加%*以将原始参数传递给主参数。main 中唯一"奇怪"的情况是%0:main而不是执行脚本。但%~f0可用于获取执行脚本的完整路径。

测试.bat

@echo off
9>&2 2>nul (call :lockAndRestoreStdErr %* 8>"%~f0.lock") && (
del "%~f0.lock"
) || (
echo Only one instance allowed - "%~f0" is already running >&2
)
exit /b
:lockAndRestoreStdErr
call :main %* 2>&9
exit /b 0
:main
echo        %%0 = %0
echo    "%%~f0" = "%~f0"
echo Arguments = %*
pause
exit /b

示例实例 1 输出

C:test>test arg1 arg2
%0 = :main
"%~f0" = "C:testtest.bat"
Arguments = arg1 arg2
Press any key to continue . . .

示例实例 2 输出将实例 1 仍在运行

C:test>test arg1 arg2
Only one instance allowed - "C:testtest.bat" is already running
C:test>

实例1 将在按下某个键后终止,然后实例 2 可以运行脚本。

编辑

实际上,您不需要单独的锁定文件。您可以使用批处理脚本本身作为锁定文件。只要记住使用>>而不是>,否则你会抹掉脚本!

@echo off
9>&2 2>nul (call :lockAndRestoreStdErr %* 8>>"%~f0") || (
echo Only one instance allowed - "%~f0" is already running >&2
)
exit /b
:lockAndRestoreStdErr
call :main %* 2>&9
exit /b 0
:main
echo        %%0 = %0
echo    "%%~f0" = "%~f0"
echo Arguments = %*
pause
exit /b

如果用户对脚本没有写入权限,则上述方法都不起作用。在这种情况下,您应该将锁定文件放在%temp%文件夹中。

@echo off
9>&2 2>nul (call :lockAndRestoreStdErr %* 8>"%temp%%~nx0.lock") && (
del "%temp%%~nx0.lock"
) || (
echo Only one instance allowed - "%~f0" is already running >&2
)
exit /b
:lockAndRestoreStdErr
call :main %* 2>&9
exit /b 0
:main
echo        %%0 = %0
echo    "%%~f0" = "%~f0"
echo Arguments = %*
pause
exit /b

编辑 - 回答评论中的问题

要将上述技术放入脚本中,只需删除任务列表行,并将其替换为我的技术,这样脚本的顶部如下所示:

@echo off
9>&2 2>nul (call :lockAndRestoreStdErr %* 8>>"%~f0") || (
echo Only one instance allowed - "%~f0" is already running >&2
pause
)
exit /b
:lockAndRestoreStdErr
call :main %* 2>&9
exit /b 0
:main
title lootbot
:: enable save files location (persistant memory)
etc...

最新更新