3

用户 cashfoley 在codeplex上为一个名为 PSClass 的“模块”发布了一组看起来相当优雅的代码。

当我将 psclass 代码点源到我自己的一些代码中时,我可以编写如下代码:

$Animal = New-PSClass Animal {

    constructor {
        param( $name, $legs )

        # ...
    }

    method -override ToString {
        "A $($this.Class.ClassName) named $($this.name) with $($this.Legs) Legs"
    }
}

但是,当我尝试使用 PSClass 代码创建模块时,我开始遇到错误。和名称不再被识别constructormethod

查看实际的实现,我看到的是constructor,method等实际上是 New-PSClass 函数内部的嵌套函数。

因此,在我看来,当我对 PSClass.ps1 文件进行点源时,我的脚本块被允许包含对嵌套在其他本地函数中的函数的引用。但是当 PSClass 代码变成一个模块时,导出了 New-PSClass 函数(我尝试使用清单和使用 Export-ModuleMember),名称不再可见。

有人可以向我解释嵌套函数的脚本块、范围规则和可见性规则如何协同工作吗?

另外,对于纯 Powershell 脚本,是否有更好的类定义协议?(具体来说,不涉及“只需用 C# 编写,然后执行此操作......”)

4

1 回答 1

0

脚本块中的变量在执行之前不会被评估。如果脚本块中的变量在执行块时在当前范围内不存在,则这些变量将没有任何值。脚本块不是闭包:它们不会在实例化时捕获上下文。

Remove-variable FooBar

function New-ScriptBlock
{
    $FooBar = 1

    $scriptBlock = {
        Write-Host "FooBar: $FooBar"
    }

    $FooBar = 2
    & $scriptBlock  # Outputs FooBar: 2 because $FooBar was set to 2 before invocation
    return $scriptBlock
}

function Invoke-ScriptBlock
{
    param(
        $ScriptBlock
    )

    & $ScriptBlock
}

$scriptBlock = New-ScriptBlock 

& $scriptBlock    # Prints nothing since $FooBar doesn't exist in this scope

$FooBar = 3
Invoke-ScriptBlock $scriptBlock   # Prints $FooBar: 3 since FooBar set to 3
于 2012-06-14T14:30:33.967 回答