trap { write-host "Error connecting to $computer" -fore red "$computer" | out-file c:demoerrors.txt -appendcontinue}$computer = "localhost"get-wmiobject win32_operatingsystem -comp $computer $computer = "server2"get-wmiobject win32_operatingsystem -comp $computer $computer = "localhost"get-wmiobject win32_operatingsystem -comp $computer此脚本的输出(如图 2 所示)与我的期望不符。请注意 "Error connecting to…" 消息不显示。也没有创建 Errors.txt 文件。也就是说,根本没有执行我的 Trap。究竟发生了什么?
trap { write-host "Error connecting to $computer" -fore red"$computer" | out-file c:demoerrors.txt -appendcontinue}$computer = "localhost"get-wmiobject win32_operatingsystem -comp $computer -ea stop$computer = "server2"get-wmiobject win32_operatingsystem -comp $computer -ea stop$computer = "localhost"get-wmiobject win32_operatingsystem -comp $computer -ea stop
图 4 使用 –ErrorAction 参数时我获得了更多有用的结果
在 Trap 末尾使用 Continue 指示外壳继续执行产生异常的代码行之后的一行。还可以使用关键字 Break(我将在稍后加以讨论)。另请注意,$computer 变量(在脚本中定义)在 Trap 内仍然有效。这是因为 Trap 是脚本本身的子作用域,即 Trap 可以查看脚本内的所有变量(稍后我也将介绍此方面的更多相关信息)。
在作用域中完成所有操作
Windows PowerShell 中错误捕获的一个尤为棘手的方面是作用域的使用。外壳本身代表全局作用域,它包含外壳内部发生的所有事件。如果您运行某个脚本,它会获取自己的脚本作用域。如果您定义某个函数,该函数的内部便是其自己的专用作用域等等。这将创建一种父/子类型的层次结构。
发生异常时,外壳会在当前作用域内查找 Trap。这意味着某个函数内的异常将在该函数内部查找 Trap。如果外壳发现了 Trap,就会执行该 Trap。如果 Trap 以 Continue 结尾,外壳将继续执行引发异常的代码行后面的一行,但仍在同一作用域中。下面借助一小部分伪代码来说明这一点:
Trap { # Log error to a file Continue}Get-WmiObject Win32_Service –comp "Server2" –ea "Stop"Get-Process如果异常发生在第 5 行,则将执行第 1 行中的 Trap。Trap 以 Continue 结尾,因此将继续执行第 6 行。
Trap {# Log error to a fileContinue }Function MyFunction {Get-WmiObject Win32_Service –comp "Server2" –ea "Stop"Get-Process }MyFunction Write-Host "Testing!"如果错误发生在第 7 行,则外壳会在函数的作用域内查找 Trap。如果没有找到,那么外壳将退出函数的作用域,继续在父作用域内查找 Trap。因为那里有 Trap,所以它将执行第 1 行。在本例中,代码是 Continue,所以将继续执行同一作用域中异常之后的代码行,即第 12 行,而不是第 8 行。换言之,外壳在退出之后不会再重新进入该函数。
Function MyFunction { Trap {# Log error to a fileContinue } Get-WmiObject Win32_Service –comp "Server2" –ea "Stop" Get-Process} MyFunctionWrite-Host "Testing!"在本例中,第 6 行中的错误将执行第 2 行中的 Trap,并保持在函数的作用域内。Continue 关键字将保持在该作用域内,继续执行第 7 行。如果您将 Trap 放入预期会发生错误的作用域内,好处是您仍保持在作用域中并可以在其中继续执行。但如果此方法对于您的情况不适用应该怎么办呢?
Get-Service | Export-CliXML c:aseline.xml几乎所有对象都可以输送到 Export-CliXML,它会将对象转换为 XML 文件。而后,您可以运行同一命令(如 Get-Service)并将结果与保存的 XML 进行比较。命令如下:
Compare-Object (Get-Service) (Import-CliXMLc:aseline.xml) –property name添加 –property 参数将强制比较仅查看该属性,而非整个对象。在本例中,您将得到由不同于原始基线的所有服务名称组成的列表,让您了解在创建后基线是否添加或删除了任何服务。
Trap {# Handle the errorContinue }Function MyFunction {Trap { # Log error to a file If ($condition) {Continue } Else {Break }}Get-WmiObject Win32_Service –comp "Server2" –ea "Stop"Get-Process }MyFunction Write-Host "Testing!"以下简要概述了执行链。首先执行第 19 行,它调用第 6 行中的函数。执行第 15 行并产生异常。该异常在第 7 行捕获,然后 Trap 必须在第 9 行做出决定。假设 $condition 为 True,Trap 将在第 16 行继续执行。