4

什么有效 -

假设我有一个scriptblockSelect-Objectcmdlet 一起使用的。

$jobTypeSelector = `
    {
        if ($_.Type -eq "Foo")
        {
            "Bar"
        }
        elseif ($_.Data -match "-Action ([a-zA-Z]+)")
        {
            $_.Type + " [" + $Matches[1] + "]"
        }
        else
        {
            $_.Type
        }
    }

$projectedData = $AllJobs | Select-Object -Property State, @{Name="Type"; Expression=$jobTypeSelector}

这工作正常,我得到了预期的结果。

我正在尝试做的事情 -

但是,在稍后的代码中,我想重用scriptblock定义为$jobTypeSelector.

例如,我希望下面的代码采用$fooJob注意它是单个对象)作为下面的参数传递,并用于$_自动变量scriptblock并返回相同的结果,因为它在Select-Objectcmdlet 的上下文中执行时返回。

$fooType = $jobTypeSelector.Invoke($fooJob)

什么不起作用 -

它不像我预期的那样工作,我得到了空值。

我已经尝试过的-

  1. 我检查了,属性都设置正确,不是因为相关属性本身是空白或$null.

  2. 我在互联网上查看,似乎很难找到关于该主题的任何相关页面,但我最终找到了一个在稍微不同的上下文中接近解释该问题的页面——在 PowerShell 中调用脚本块。该博客没有直接回答我的问题,任何相关的解释只会导致一个非常丑陋的解决方案,在我看来很难阅读和维护。

问题 -

那么,scriptblock单个对象调用 a 的最佳方法是什么,它使用$_自动变量作为参数(而不是param块)

4

2 回答 2

5

在摆弄了各种选项之后,我最终以一种 Hackish 的方式解决了这个问题。但我发现它是最好的解决方案,因为它体积小,易于阅读、维护和理解。

即使我们谈论的是单个对象,也可以通过cmdlet在管道中使用它(即PowerShell定义$_自动变量时)ForEach-Object

$fooType = $fooJob | ForEach-Object $jobTypeSelector
于 2014-11-25T21:12:10.220 回答
2

你当然可以使用foreach或者ForEach-Object你提到的。

如果您通过设置将其从函数 ScriptBlock 更改为过滤器 ScriptBlock,您也可以直接通过管道连接到ScriptBlockIsFilter$true

$jobTypeSelector.IsFilter = $true

$fooType = $fooJob | $jobTypeSelector

但是,如果您使用命名函数而不是匿名 ScriptBlock 会更好,例如:

function Get-JobType
{
    Param ( 
        [object] $Job 
    )

    if ($Job.Type -eq "Foo")
    {
        "Bar"
    }
    elseif ($Job.Data -match "-Action ([a-zA-Z]+)")
    {
        $Job.Type + " [" + $Matches[1] + "]"
    }
    else
    {
        $Job.Type
    }
}

Select-Object然后你可以像这样使用它select

$projectedData = $AllJobs |
    select -Property State, @{Name="Type"; Expression={Get-JobType $_}}

或者只有一个工作,像这样:

$fooType = Get-JobType $fooJob
于 2015-05-20T06:15:32.913 回答