9

我在使用 Powershell 时遇到了一个特殊问题。我在 catch 块中捕获了一个异常,但没有填充全局 $Error 对象。

一个简单的例子,它会按预期运行是这样的:

function Bar
{
  Foo
}

function Foo
{
  try
  {
    $Error.Clear()
    throw "Error!"
  }
  catch
  {
    "Caught an error - current error count $($Error.Count)"  
  } 
  finally
  {
    "Cleaning up - current error count $($Error.Count)"  
  }
}

如果您调用 Bar,输出与您期望的一样

Caught an error - current error count 1
Cleaning up - current error count 1

我遇到问题的代码几乎相同,只是它从模块加载 Foo 。不确定这是一个错误,还是我不明白的东西(必须检查我的 Powershell in Action 书!)

如果我将 Foo 保存到一个模块 - Foo.psm1

function Foo
{
  try
  {
    $Error.Clear()
    throw "Error!"
  }
  catch
  {
    "Caught an error - current error count $($Error.Count)"  
  } 
  finally
  {
    "Cleaning up - current error count $($Error.Count)"  
  }
}

Export-ModuleMember -Function Foo

然后我执行以下操作

Import-Module .\Foo.psm1
$Error.Clear()
"Current error count $($Error.Count)"
Foo
"Current error count $($Error.Count)"

我最终得到

Current error count 0
Caught an error - current error count 0
Cleaning up - current error count 0
Current error count 1

请注意,Foo 不再看到对 $Error 所做的任何更改。因此,代码的模块化正在改变错误传播行为。任何人都可以加入这背后的推理吗?

我应该注意,我可以通过自动变量 $_ 获取特定捕获的异常,但我希望在调用堆栈中的这一点上获得整个集合。

4

2 回答 2

13

正如 Ethan 所提到的,您需要查看 $global:error 以查看所有错误。模块总是有它们自己的 $error 变量,但它(大部分)没有被使用。

背景:有一次,我们曾考虑将模块代码中发生的错误隔离到模块范围,但最终决定将所有错误添加到全局错误集合中,因为它本质上是一个错误日志(如内存中的事件日志)。不幸的是模块作用域 $error 在发布之前没有被删除,因此您需要使用全局作用域限定符来访问“真实”$error 变量。

布鲁斯·帕耶特,微软公司

于 2012-07-12T17:48:46.473 回答
10

我之前没有注意到这一点,但也许 $error 集合像任何其他变量一样被限定为模块。尝试在测试脚本的关键点比较以下两个显式范围变量的值:

"Errors - Global: {0}; Module: {1}" -f $global:error.count, $script:error.count

让我知道你是怎么办的。

于 2012-07-12T01:20:43.077 回答