gpt4 book ai didi

PowerShell: `$matches` 是否保证与管道变量同步执行管道?

转载 作者:行者123 更新时间:2023-12-03 00:13:42 24 4
gpt4 key购买 nike

首先,制作一些示例文件:

2010..2015 | % { "" | Set-Content "example $_.txt" }

#example 2010.txt
#example 2011.txt
#example 2012.txt
#example 2013.txt
#example 2014.txt
#example 2015.txt

我想要做的是将年份与正则表达式捕获组匹配,然后使用 $matches[1] 引用匹配并使用它。我可以在一个脚本 block 和一个 cmdlet 中编写此代码,并且可以正常工作:
gci *.txt | foreach { 

if ($_ -match '(\d+)') # regex match the year
{ # on the current loop variable
$matches[1] # and use the capture group immediately
}

}
#2010
#2011
#.. etc

我也可以写这个在一个脚本 block 中进行匹配,然后引用 $matches稍后在另一个 cmdlet 的脚本 block 中:
gci *.txt | where { 

$_ -match '(\d+)' # regex match here, in the Where scriptblock

} | foreach { # pipeline!

$matches[1] # use $matches which was set in the previous
# scriptblock, in a different cmdlet
}

它具有相同的输出,并且似乎工作正常。但它是否保证有效,还是未定义和巧合?

可以 'example 2012.txt'得到匹配,然后缓冲。 'example 2013.txt'匹配,然后缓冲。 | foreach开始工作 'example 2012.txt'但是 $matches已更新为 2013他们不同步?

我不能让它们不同步——但我仍然可以依赖未定义的行为。

(FWIW,为了清晰和可读性,我也更喜欢第一种方法)。

最佳答案

本身没有同步进行。由于管道的工作方式,第二个示例有效。随着每个单个对象通过满足 Where-Object 中的条件传递, -Process阻止 ForEach-Object立即处理它,所以 $Matches尚未被任何其他 -match 覆盖手术。

如果您要做一些导致管道在传递对象之前收集对象的操作,例如排序,那么您将遇到麻烦:

gci *.txt | where { 

$_ -match '(\d+)' # regex match here, in the Where scriptblock

} | sort | foreach { # pipeline!

$matches[1] # use $matches which was set in the previous
# scriptblock, in a different cmdlet
}

例如,上面应该失败,输出 n 个对象,但它们都将是最后一个匹配项。

所以谨慎的做法是不要依赖它,因为它掩盖了危险。其他人(或几个月后的您)可能不会想到插入 sort然后对结果感到非常困惑。

TheMadTechnician在评论中指出,位置改变了事情。将排序放在您引用 $Matches 的部分之后(在 foreach 中),或在使用 where 过滤之前,它仍然会按预期工作。

我认为这让我们明白应该避免它,因为它相当不清楚。如果代码在您无法控制的部分管道中发生更改,则行为可能会意外地不同。

我有时喜欢抛出一些冗长的输出来证明这一点:

原来的
gci *.txt | where { 
"Where-Object: $_" | Write-Verbose -Verbose
$_ -match '(\d+)' # regex match here, in the Where scriptblock

} | foreach { # pipeline!
"ForEach-Object: $_" | Write-Verbose -Verbose
$matches[1] # use $matches which was set in the previous
# scriptblock, in a different cmdlet
}

已排序
gci *.txt | where { 
"Where-Object: $_" | Write-Verbose -Verbose
$_ -match '(\d+)' # regex match here, in the Where scriptblock

} | sort | foreach { # pipeline!
"ForEach-Object: $_" | Write-Verbose -Verbose
$matches[1] # use $matches which was set in the previous
# scriptblock, in a different cmdlet
}

您将看到的不同之处在于,在原版中,只要 where “清除”一个对象, foreach马上得到它。在排序后,你可以看到所有的 where s 首先发生,在 foreach 之前得到其中任何一个。
sort没有任何详细的输出,所以我没有费心那样称呼它,但本质上是它的 Process {} block 只是收集所有对象,以便它可以比较(排序!)它们,然后将它们吐出 End {}堵塞。

更多示例

首先,这是一个模拟 Sort-Object 的函数。的对象集合(它实际上并不对它们进行排序或做任何事情):
function mocksort {
[CmdletBinding()]
param(
[Parameter(
ValueFromPipeline
)]
[Object]
$O
)

Begin {
Write-Verbose "Begin (mocksort)"

$objects = @()
}

Process {
Write-Verbose "Process (mocksort): $O (nothing passed, collecting...)"

$objects += $O
}

End {
Write-Verbose "End (mocksort): returning objects"

$objects
}
}

然后,我们可以将其与前面的示例一起使用,并在最后进行一些 sleep :
gci *.txt | where { 
"Where-Object: $_" | Write-Verbose -Verbose
$_ -match '(\d+)' # regex match here, in the Where scriptblock

} | mocksort -Verbose | foreach { # pipeline!
"ForEach-Object: $_" | Write-Verbose -Verbose
$matches[1] # use $matches which was set in the previous
# scriptblock, in a different cmdlet
} | % { sleep -milli 500 ; $_ }

关于PowerShell: `$matches` 是否保证与管道变量同步执行管道?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42729506/

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