我正在尝试编写一些 PowerShell 函数来做一些事情,然后透明地调用现有的内置函数。我想原封不动地传递所有论点。我不想知道论点的任何细节。
我厌倦了使用'splat'来做到这一点,@args
但这并没有像我预期的那样工作。
在下面的示例中,我编写了一个名为的玩具函数myls
,它应该打印hello! 然后调用相同的内置函数 ,Get-ChildItem
该内置别名ls
调用的参数行的其余部分保持不变。到目前为止,我所拥有的效果很好:
function myls
{
Write-Output "hello!"
# $MyInvocation | Format-List # <-- uncomment this line for debug info
Invoke-Expression ("Get-ChildItem " + $MyInvocation.UnboundArguments -join " ")
}
的正确版本myls
应该能够处理不带参数、带一个参数、带命名参数、从包含多个分号分隔的命令的行以及参数中的变量(包括包含空格的字符串变量)调用。基本上,它应该是ls
.
下面的测试比较myls
和内置ls
:
[注意:输出被省略和/或压缩以节省空间]
PS> md C:\p\d\x, C:\p\d\y, C:\p\d\"jay z"
PS> cd C:\p\d
PS> ls # no args
PS> myls # pass
PS> cd ..
PS> ls d # one arg
PS> myls d # pass
PS> $a="A"; $z="Z"; $y="y"; $jz="jay z"
PS> $a; ls d; $z # multiple statements
PS> $a; myls d; $z # pass
PS> $a; ls d -Exclude x; $z # named args
PS> $a; myls d -Exclude x; $z # pass
PS> $a; ls d -Exclude $y; $z # variables in arg-line
PS> $a; myls d -Exclude $y; $z # pass
PS> $a; ls d -Exclude $jz; $z # variables containing spaces in arg-line
PS> $a; myls d -Exclude $jz; $z # FAIL!
有没有办法可以重写myls
以获得我想要的行为?
简短的回答:是的,这是可能的。坏消息:它需要知道参数细节的代码和关于一个希望调用的函数的其他元数据。好消息:不需要自己写这些。此元数据以编程方式可用,并且存在可用于自动生成骨架代理代码的模块(请参阅下面的@Jaykul 的答案)。我选择使用名为 "MetaProgramming" 的模块。导入后,生成一个插入式myls
脚本非常简单:
New-ProxyCommand ls > .\myls.ps1
然后可以开始自定义新生成的myls.ps1
脚本,如下所示:
...
begin
{
Write-Output "hello!" # <-- add this line
try {
$outBuffer = $null
...
瞧!这个新版本通过了所有测试。