在批处理文件的参数中转义“、<、>、>> 或 | 等字符

Escaping characters like ", <, >, >>, or | in the arguments to a batch file

正在尝试做:

fake-command.bat "ping -n 4 -w 1 127.0.0.1 >NUL"

fake-command.bat ping -n 4 -w 1 127.0.0.1

批处理文件可能如下所示:

@echo %*

应该return:

ping -n 4 -w 1 127.0.0.1 >NUL

ping -n 4 -w 1 127.0.0.1

这里有一个解决方法:

@echo off
goto start
------------------------------------------------------

Usage : mystring <command>

  Quotes around the command are required only when the
  command involves redirection via <, >, >>, or |, etc.
  Quotes ensure that the redirection is applied to the 
  command, rather than the bat command itself.

Examples :

  mystring ping -n 4 -w 1 127.0.0.1
  mystring "ping -n 4 -w 1 127.0.0.1 >NUL"

------------------------------------------------------
:start
SETLOCAL ENABLEDELAYEDEXPANSION

SET "MYSTRING=%*"
ECHO My String = !MYSTRING!
SET !MYSTRING=MYSTRING:>=^>!

CALL :BATCH_FUNCTION !MYSTRING!
GOTO :EOF

:BATCH_FUNCTION
SET "ARGS=%~1"
ECHO Arguments = !ARGS!
endlocal
GOTO :EOF

问题是,之后:mystring "ping -n 1 127.0.0.1 >NUL"

它returns:

My String =
Arguments =

及之后:mystring ping -n 1 127.0.0.1

它returns:

My String = ping -n 1 127.0.0.1
Arguments = ping

更新: 我用 Get list of passed arguments in Windows batch script (.bat)

中的以下代码更新了问题
@echo off
SETLOCAL DisableDelayedExpansion
SETLOCAL
if exist param.txt (del param.txt)
for %%a in ('%*') do (
    set "prompt="
    echo on
    for %%b in ('%*') do rem * #%~1#
    @echo off
) > param.txt
ENDLOCAL

for /F "delims=" %%L in (param.txt) do (
  set "param1=%%L"
)
SETLOCAL EnableDelayedExpansion
set "param1=!param1:*#=!"
set "param1=!param1:~0,-2!"
echo My string is = !param1!

使用这段代码,我可以获得转义字符,但如果参数不在引号中,输出就会损坏

什么有效?

mystring "# % $ ` ' ( ) < << >> > >NUL && & || | { } \ / - + = , . : ; ^ " &REM OK
mystring "ping -n 4 -w 1 127.0.0.1 >NUL" &REM OK

它returns:

My string is = # % $ ` ' ( ) < << >> > >NUL && & || | { } \ / - + = , . : ; ^
My string is = ping -n 4 -w 1 127.0.0.1 >NUL 

什么不起作用?

mystring ping -n 4 -w 1 127.0.0.1 &REM NOK
mystring "*" &REM NOK

它returns:

My string is = ping
The system can not find the file param.txt.
My string is = *

我看不出如何在此代码中添加

同样适用于单个参数"ping -n 4 -w 1 127.0.0.1 >NUL"的批处理代码是:

@echo off
goto start
------------------------------------------------------

Usage : mystring <command>

  Quotes around the command are required only when the
  command involves redirection via <, >, >>, or |, etc.
  Quotes ensure that the redirection is applied to the
  command, rather than the bat command itself.

Examples :

  mystring ping -n 4 -w 1 127.0.0.1
  mystring "ping -n 4 -w 1 127.0.0.1 >NUL"

------------------------------------------------------
:start
SETLOCAL ENABLEDELAYEDEXPANSION

if "%~2" == "" (SET "MYSTRING=%~1") else (SET "MYSTRING=%*")
ECHO My String = !MYSTRING!
SET "!MYSTRING=MYSTRING:>=^>!"

CALL :BATCH_FUNCTION "!MYSTRING!"
GOTO :EOF

:BATCH_FUNCTION
SET "ARGS=%~1"
ECHO Arguments = !ARGS!
endlocal
GOTO :EOF

开头的 IF 条件产生了重要的差异。

当批处理文件被调用时

fake-command.bat ping -n 4 -w 1 127.0.0.1

它使用 6 个参数调用,因此 %* 是将所有参数分配给变量 MYSTRING 的正确方法。

但是用

调用批处理文件
fake-command.bat "ping -n 4 -w 1 127.0.0.1 >NUL"

表示只有 1 个用双引号引起来的参数被传递给批处理文件。

SET "MYSTRING=%*"

导致将字符串

分配给变量MYSTRING
"ping -n 4 -w 1 127.0.0.1 >NUL"

字符串中包含引号。剩余的引号会导致进一步处理的错误结果。

尚不清楚应将什么分配给变量 ARGS。如果传递给批处理文件的所有参数都应分配给此变量,则在调用子例程时也需要用双引号将 !MYSTRING! 括起来,以将批处理文件的所有参数作为 1 个参数传递给子例程。

命令 FOR 可用于将分配给 MYSTRING 的字符串拆分为命令 (ping) 及其参数(使用重定向器运算符)。


UPDATE(问题更新后):

根本不清楚任务是什么。 This batch code 旨在仅获取传递给批处理文件的第一个参数,这是 #%1# 在第二个内部 FOR 循环中的原因。这正是批处理代码所做的,尽管它包含一个对结果没有影响的小错误。

set "prompt=" 没有效果,因为不可能将提示文本更改为无。所以文件 params.txt 仍然包含提示文本。更好的方法是使用 prompt $_set "prompt=$_" ,它们将提示文本定义为仅回车 return + 换行,因此只生成一个空行,稍后使用的命令 FOR 在读取生成的文件时 params.txt。在第一个外部 FOR 循环之后,使用 endlocal 恢复先前的提示定义。

Paul,您只是将 (1) 替换为 ('%*'),这只会影响传递给批处理文件的每个参数,第一个且只有第一个参数被写入文本文件 params.txt 然后再读几次。只需查看 params.txt 即可查看 运行 具有各种参数的批处理文件之后包含的内容。这当然没什么用。

这里是修改后的批处理代码,输出传递给批处理文件的所有参数。

@echo off
setlocal DisableDelayedExpansion
del param.txt 2>nul
for %%a in (1) do (
    set "prompt=$_"
    echo on
    for %%b in (%*) do rem * #%%~b#
    @echo off
) > param.txt
endlocal

setlocal EnableDelayedExpansion
set "ParaNumber=1"
set "AllParameters="
for /F "delims=" %%L in (param.txt) do (
    set "Parameter=%%L"
    set "Parameter=!Parameter:*#=!"
    set "Parameter=!Parameter:~0,-2!"
    set "AllParameters=!AllParameters! !Parameter!"
    echo Parameter !ParaNumber! = !Parameter!
    set /A ParaNumber+=1
)
set "AllParameters=!AllParameters:~1!"
echo All parameters: !AllParameters!
endlocal

下面三个字符串依次调用批处理文件三次

"# % $ ` ' ( ) < << >> > >NUL && & || | { } \ / - + = , . : ; ^ "
"ping -n 4 -w 1 127.0.0.1 >NUL"
ping -n 4 -w 1 127.0.0.1

产生以下三个输出

Parameter 1 = # % $ ` ' ( ) < << >> > >NUL && & || | { } \ / - + = , . : ; ^
All parameters: # % $ ` ' ( ) < << >> > >NUL && & || | { } \ / - + = , . : ; ^

Parameter 1 = ping -n 4 -w 1 127.0.0.1 >NUL
All parameters: ping -n 4 -w 1 127.0.0.1 >NUL

Parameter 1 = ping
Parameter 2 = -n
Parameter 3 = 4
Parameter 4 = -w
Parameter 5 = 1
Parameter 6 = 127.0.0.1
All parameters: ping -n 4 -w 1 127.0.0.1 >NUL

使用 "*" 调用批处理文件现在会在当前目录中将所有非隐藏文件列为参数 1 到 n。因此,此批处理文件的修改版本不适用于有效文件名模式的参数。