如果我有一个接受多个字符串参数的函数,第一个参数似乎获得分配给它的所有数据,其余参数作为空传入。
一个快速测试脚本:
Function Test([string]$arg1, [string]$arg2)
{
Write-Host "`$arg1 value: $arg1"
Write-Host "`$arg2 value: $arg2"
}
Test("ABC", "DEF")
生成的输出为
$arg1 value: ABC DEF
$arg2 value:
正确的输出应该是:
$arg1 value: ABC
$arg2 value: DEF
这似乎在多台机器上v1和v2之间是一致的,所以很明显,我做错了什么。有人能指出具体是什么吗?
因为这是一个常见的问题,所以我想提一下,PowerShell函数应该使用认可的动词(动词-名词作为函数名)。
名称的动词部分标识cmdlet执行的操作。名称的名词部分标识执行动作的实体。该规则简化了高级PowerShell用户对cmdlet的使用。
此外,你还可以指定参数是否为强制参数以及参数的位置:
function Test-Script
{
[CmdletBinding()]
Param
(
[Parameter(Mandatory=$true, Position=0)]
[string]$arg1,
[Parameter(Mandatory=$true, Position=1)]
[string]$arg2
)
Write-Host "`$arg1 value: $arg1"
Write-Host "`$arg2 value: $arg2"
}
要将参数传递给函数,你可以使用position:
Test-Script "Hello" "World"
或者指定参数名:
Test-Script -arg1 "Hello" -arg2 "World"
你不像在c#中调用函数那样使用圆括号。
我建议在使用多个参数时始终传递参数名,因为这样可读性更好。
如果你不知道(或关心)你将传递多少参数给函数,你也可以使用一个非常简单的方法,如;
代码:
function FunctionName()
{
Write-Host $args
}
这将打印出所有的参数。例如:
FunctionName a b c 1 2 3
输出
a b c 1 2 3
我发现这在创建使用外部命令的函数时特别有用,这些外部命令可能有许多不同的(可选的)参数,但依赖于所述命令提供关于语法错误等的反馈。
下面是另一个真实世界的例子(为tracert命令创建一个函数,我讨厌不得不记住截断的名称);
代码:
Function traceroute
{
Start-Process -FilePath "$env:systemroot\system32\tracert.exe" -ArgumentList $args -NoNewWindow
}
我没有看到这里提到它,但是分散参数是一种有用的替代方法,如果您动态地构建命令的参数(而不是使用Invoke-Expression),它会变得特别有用。你可以用数组来表示位置参数,用哈希表来表示命名参数。下面是一些例子:
注意:您可以相对轻松地将位置splats与外部命令参数一起使用,但命名splats与外部命令使用起来不太有用。它们可以工作,但程序必须接受-Key:Value格式的参数,因为每个参数都与哈希表的键/值对相关。这类软件的一个例子是来自Windows的Chocolatey包管理器的choco命令。
Splat With Arrays(位置参数)
使用位置参数测试连接
Test-Connection www.google.com localhost
与阵列溅射
$argumentArray = 'www.google.com', 'localhost'
Test-Connection @argumentArray
注意,在splapping时,我们用@而不是$引用splapping变量。当使用Hashtable进行splat时也是一样的。
Splat With Hashtable (Named Arguments)
使用命名参数测试连接
Test-Connection -ComputerName www.google.com -Source localhost
用哈希表喷溅
$argumentHash = @{
ComputerName = 'www.google.com'
Source = 'localhost'
}
Test-Connection @argumentHash
同时拆分位置参数和命名参数
使用位置参数和命名参数测试连接
Test-Connection www.google.com localhost -Count 1
飞溅数组和哈希表一起
$argumentHash = @{
Count = 1
}
$argumentArray = 'www.google.com', 'localhost'
Test-Connection @argumentHash @argumentArray