gpt4 book ai didi

powershell - 为什么从 powershell 调用批处理文件时插入符会消失?

转载 作者:行者123 更新时间:2023-12-02 08:05:51 29 4
gpt4 key购买 nike

这是我的 args.bat 文件:

@echo off
python -c "import sys; print(sys.argv)" %*

我在这里调用 python 因为我确切地知道如何解释输出,而不是猜测字符串中有哪些引号

如果我从 cmd 调用它会发生以下情况:

>.\args.bat ^^^^^
More?
More?
['-c', '^']
>.\args.bat "^^^^^"
['-c', '^^^^^']

但这是 powershell 的作用:

PS> .\args.bat ^^^^^
['-c', '^']
PS> .\args.bat "^^^^^"
['-c', '^']
PS> .\args.bat '^^^^^'
['-c', '^']
PS> .\args.bat "'^^^^^'"
['-c', '^']
PS> .\args.bat '"^^^^^"'
['-c', '^^^^^']

为什么在从 powershell 调用批处理文件时必须双引号参数?


从powershell调用python不存在同样的问题:

PS> python -c "import sys; print(sys.argv)" "^^^^^"
['-c', '^^^^^']
PS> python -c "import sys; print(sys.argv)" '^^^^^'
['-c', '^^^^^']
PS> python -c "import sys; print(sys.argv)" ^^^^^
['-c', '^^^^^']

但是从 powershell 从 python 调用批处理时仍然存在!

PS> python -c "import subprocess; subprocess.call(['args.bat', '^^^^^'])"
['-c', '^']

然后从 cmd 从 python 批处理:

> python -c "import subprocess; subprocess.call(['args.bat', '^^^^^'])"
['-c', '^']

最佳答案

关于批处理文件是否可以稳健地传递任意参数(剧透警报:否)的更大问题,请参阅 Eric 的 follow-up question .


tl;dr:

PS> .\args.bat --% "^^^^^"
['-c', '^^^^^']

使用 --%, the stop-parsing symbol , 使 PowerShell 按原样传递剩余的参数,除了扩展 cmd.exe 样式的环境变量引用,例如 %OS%

相反,您将无法引用 PowerShell 变量或表达式。


PowerShell 作为一个独立的 shell,首先执行其自己的解析,然后 - Windows 上的一个不幸的必需品 - 在为外部程序编写命令行时重新应用引号

它执行此操作的方式随着时间的推移而发生了变化,并且在过去一直是许多问题的根源。这些问题导致了 --% 的引入,如上所述,但它也有其自身的挑战。

然而,重新引用通常是必要的,因为外部程序不一定理解 PowerShell 的语法。
最值得注意的是,'...' 引用不能假定为所有外部程序都能理解,因此需要转换为 "..."

但是,PowerShell 会根据它是否认为必要有条件地 重新引用。

在本例中,因为 PowerShell "^^^^^" 字符串的内容 没有空格,PowerShell 在将它传递给批处理文件时决定不引用它;即:

.\args.bat "^^^^^"

有效地成为

.\args.bat ^^^^^

这对大多数外部程序来说很好,其中^没有句法意义,但对于批处理文件(由cmd.exe) 它很重要,因为 ^ 用作转义字符。在不带引号的字符串中。

要在这种情况下仍然强制使用双引号,作为上述 --% 方法的替代方法,您可以使用嵌入双引号,因为您已经在您的问题中证明:

PS> .\args.bat '"^^^^^"' #  '...' quoting with embedded "..." quoting
['-c', '^^^^^']

The same problem doesn't exist calling python from powershell:
PS> python -c "import sys; print(sys.argv)" "^^^^^"

那是因为 cmd.exe 不涉及此场景,即使从 PowerShell 调用时封闭的双引号丢失,Python 也会将 ^ 实例视为作为文字。

But persists when calling batch from python from powershell
PS> python -c "import subprocess; subprocess.call(['args.bat', '^^^^^'])"

And batch from python from cmd:> python -c "import subprocess; subprocess.call(['args.bat', '^^^^^'])"

那是因为你再次调用一个批处理文件 - 未加引号 ^^^^^ 在这种情况下甚至来自 cmd.exe,因为封闭的 '....' 仅对 Python 具有句法意义,不会被传递 - 这重新引入了特殊解释^

请注意,即使是调用批处理文件的“无外壳”方法(例如来自 Python 的 subprocess.call())仍然总是调用 cmd.exe,因为 cmd.exe 是执行批处理文件所需的解释器。

未加引号 的字符串^^^^^ 传递给批处理文件时,似乎有两轮解释:首先,作为参数的一部分解析时,每个^^ 对都变成一个^,任何未配对的^ 都将被丢弃。 在批处理文件中,因此您会看到 ^^,当不加引号使用时,它变成单个 ^ - 如果您双引号 %*%1 在批处理文件中,您会看到 ^^ 已通过。

关于powershell - 为什么从 powershell 调用批处理文件时插入符会消失?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51757727/

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