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

标签 windows batch-file escaping

尝试做:

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 %*

它应该返回:

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"

它返回:

My String =
Arguments =

及之后:mystring ping -n 1 127.0.0.1

它返回:

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

它返回:

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

它返回:

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

我看不出如何在此代码中添加 the trick from Mofi .

最佳答案

批处理代码也适用于单个参数“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) 及其参数(使用重定向器运算符)。


更新(问题更新后):

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

set "prompt=" 没有效果,因为不可能将提示文本更改为无。所以文件 params.txt 仍然包含提示文本。更好的方法是使用 prompt $_set "prompt=$_" ,它定义提示文本只是回车 + 换行,因此只产生一个空行,稍后会忽略使用命令 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。因此,此批处理文件的修改版本不适用于有效文件名模式的参数。

关于windows - 在批处理文件的参数中转义“、<、>、>> 或 | 等字符,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32414436/

相关文章:

c++ - 如何使用 Visual Studio 为 windows 编译 GMP

c - Hook Windows 登录/注销事件

android - 使用批处理命令搜索文本文件并删除字符串

Java - 转义一个字符

bash - 在不重新启动 xterm 的情况下重新加载 .Xresources

windows - SEH 异常代码中第 31 位和第 30 位的含义是什么?

c# - 丑陋的线条和边框渲染

node.js - Inno Setup - 在从安装程序执行的批处理文件中无法识别由子安装程序添加到 PATH 的 npm 程序

windows - 如何根据文件夹名称识别最新的应用程序构建文件夹

Bash - 转义单引号