PowerShell嵌套函数中的可变范围

时间:2015-04-23 18:43:03

标签: powershell scope nested-function

我从一家主要/声誉良好的公司获得了这个广泛的PowerShell脚本,该公司应该完美无瑕地工作。嗯,事实并非如此 该脚本由许多嵌套函数组成,其中许多变量传递给主父函数,然后传递给它的所有子函数。使用和修改所有这些变量的儿童。

为什么所有这些变量都不包含正确的数据? 这是我正在谈论的结构:

f1 {
     f2 {
          v #prints 0
          v = 1
          f3
     }
     f3 {
          v #prints 1
          v = 2
     }
     v = 0
     f2
     v #should print 2 but prints 0
}

1 个答案:

答案 0 :(得分:12)

在嵌套函数中,所有子函数都可以访问所有父函数'变量。对变量的任何更改都在当前函数的本地范围内可见,并且之后调用所有嵌套的子函数。子函数执行完毕后,变量将在调用子函数之前返回原始值。

为了在所有嵌套函数范围中应用变量更改,需要将变量范围类型更改为AllScope

Set-Variable -Name varName -Option AllScope

这样,无论嵌套函数中的哪个级别变量被修改,即使子函数终止并且父级将看到新的更新值,更改也是持久的。

嵌套函数中变量作用域的正常行为:

function f1 ($f1v1 , $f1v2 )
{
        function f2 ()
       {
               $f2v = 2
               $f1v1 = $f2v #local modification visible within this scope and to all its children
               f3
               "f2 -> f1v2 -- " + $f1v2 #f3's change is not visible here
       }
        function f3 ()
       {
               "f3 -> f1v1 -- " + $f1v1 #value reflects the change from f2
               $f3v = 3
               $f1v2 = $f3v #local assignment will not be visible to f2
               "f3 -> f1v2 -- " + $f1v2
       }

        f2
        "f1 -> f1v1 -- " + $f1v1 #the changes from f2 are not visible
        "f1 -> f1v2 -- " + $f1v2 #the changes from f3 are not visible
}

f1 1 0

打印输出:

f3 -> f1v1 -- 2
f3 -> f1v2 -- 3
f2 -> f1v2 -- 0
f1 -> f1v1 -- 1
f1 -> f1v2 -- 0

具有AllScope变量的嵌套函数:

function f1($f1v1, $f1v2)
{
    Set-Variable -Name f1v1,f1v2 -Option AllScope
    function f2()
    {
        $f2v = 2
        $f1v1 = $f2v #modification visible throughout all nested functions
        f3
        "f2 -> f1v2 -- " + $f1v2 #f3's change is visible here
    }
    function f3()
    {
        "f3 -> f1v1 -- " + $f1v1 #value reflects the change from f2
        $f3v = 3
        $f1v2 = $f3v #assignment visible throughout all nested functions
        "f3 -> f1v2 -- " + $f1v2 
    }

    f2
    "f1 -> f1v1 -- " + $f1v1 #reflects the changes from f2 
    "f1 -> f1v2 -- " + $f1v2 #reflects the changes from f3 
}

f1 1 0

打印输出:

f3 -> f1v1 -- 2
f3 -> f1v2 -- 3
f2 -> f1v2 -- 3
f1 -> f1v1 -- 2
f1 -> f1v2 -- 3