about_Scripts

简短说明

介绍如何在 PowerShell 中运行和编写脚本。

长说明

脚本是包含一个或多个 PowerShell 命令的纯文本文件。 PowerShell 脚本具有 .ps1 文件扩展名。

运行脚本与运行 cmdlet 非常类似。 键入脚本的路径和文件名,然后使用参数提交数据和设置选项。 可以在你的计算机上运行脚本,也可以在另一台计算机上的远程会话中运行脚本。

编写脚本可以保存命令以供以后使用,并且可以轻松与他人共享。 最重要的是,只需输入脚本路径和文件名即可运行命令。 脚本可以像文件中的单个命令一样简单,也可以像复杂的程序一样广泛。

脚本具有附加功能,例如 #Requires 特殊注释、参数的使用、对数据部分的支持以及用于安全的数字签名。 还可以为脚本以及脚本中的任何函数编写帮助主题。

如何运行脚本

在 Windows 上运行脚本之前,需要更改默认的 PowerShell 执行策略。 执行策略不适用于在非 Windows 平台上运行的 PowerShell。

默认执行策略 Restricted会阻止所有脚本运行,包括在本地计算机上编写的脚本。 有关更多信息,请参阅 about_Execution_Policies

执行策略保存在注册表中,因此只需在每台计算机上更改一次。

若要更改执行策略,请遵循以下过程。

在命令提示符处,键入:

Set-ExecutionPolicy AllSigned

Set-ExecutionPolicy RemoteSigned

更改立即生效。

若要运行脚本,请键入脚本文件的全名和完整路径。

例如,若要在 C:\Scripts 目录运行 Get-ServiceLog.ps1 脚本,请键入:

C:\Scripts\Get-ServiceLog.ps1

若要在当前目录中运行脚本,请键入当前目录的路径,或使用点来表示当前目录,后跟路径反斜杠 (.\)。

例如,若要在本地目录中运行 ServicesLog.ps1 脚本,请键入:

.\Get-ServiceLog.ps1

如果脚本具有参数,请在脚本文件名后面输入参数和参数值。

例如,以下命令使用 Get-ServiceLog 脚本的 ServiceName 参数来请求 WinRM 服务活动的日志。

.\Get-ServiceLog.ps1 -ServiceName WinRM

作为一项安全功能,双击文件资源管理器中的脚本图标或键入不带完整路径的脚本名称时,PowerShell 不会运行脚本,即使脚本位于当前目录中也是如此。 有关在 PowerShell 中运行命令和脚本的更多信息,请参阅 about_Command_Precedence

使用 PowerShell 运行

从 PowerShell 3.0 开始,可以从文件资源管理器运行脚本。

若要使用“使用 PowerShell 运行”功能,请执行以下操作:

运行文件资源管理器,右键单击脚本文件名,然后选择“使用 PowerShell 运行”。

“使用 PowerShell 运行”功能旨在运行没有必需参数且不会将输出返回到命令提示符的脚本。

有关详细信息,请参阅 about_Run_With_PowerShell

在其他计算机上运行脚本

若要在一台或多台远程计算机上运行脚本,请使用 Invoke-Command cmdlet 的 FilePath 参数。

输入脚本的路径和文件名作为 FilePath 参数的值。 脚本必须位于本地计算机上或者本地计算机能够访问的目录中。

以下命令在名为 Server01 和 Server02 的远程计算机上运行 Get-ServiceLog.ps1 脚本。

Invoke-Command -ComputerName Server01,Server02 -FilePath `
  C:\Scripts\Get-ServiceLog.ps1

获取脚本帮助

Get-Help cmdlet 可获取脚本以及 cmdlet 和其他类型命令的帮助主题。 若要获取脚本的帮助主题,请键入 Get-Help,后跟脚本的路径和文件名。 如果脚本路径位于 Path 环境变量中,则可以省略该路径。

例如,若要获取 ServicesLog.ps1 脚本的帮助,请键入:

get-help C:\admin\scripts\ServicesLog.ps1

如何编写脚本

脚本可以包含任何有效的 PowerShell 命令,包括单个命令、使用管道、函数和控制结构(例如 If 语句和 For 循环)的命令。

要编写脚本,请在文本编辑器中打开一个新文件,键入命令,然后将它们保存在具有有效文件名和 .ps1 文件扩展名的文件中。

以下示例是一个简单的脚本,用于获取当前系统上运行的服务并将其保存到日志文件中。 日志文件名是从当前日期创建的。

$date = (get-date).dayofyear
get-service | out-file "$date.log"

若要创建此脚本,请打开文本编辑器或脚本编辑器,键入这些命令,然后将其保存在名为 ServiceLog.ps1 的文件中。

脚本中的参数

若要在脚本中定义参数,请使用 Param 语句。 Param 语句必须是脚本中的第一个语句,注释和任何 #Require 语句除外。

脚本参数的工作方式与函数参数类似。 参数值可用于脚本中的所有命令。 函数参数的所有功能,包括参数属性及其命名参数,在脚本中也有效。

运行脚本时,脚本用户在脚本名称之后键入参数。

以下示例显示了具有 ComputerName 参数的 Test-Remote.ps1 脚本。 这两个脚本函数都可以访问 ComputerName 参数值。

param ($ComputerName = $(throw "ComputerName parameter is required."))

function CanPing {
   $error.clear()
   $tmp = test-connection $computername -erroraction SilentlyContinue

   if (!$?)
       {write-host "Ping failed: $ComputerName."; return $false}
   else
       {write-host "Ping succeeded: $ComputerName"; return $true}
}

function CanRemote {
    $s = new-pssession $computername -erroraction SilentlyContinue

    if ($s -is [System.Management.Automation.Runspaces.PSSession])
        {write-host "Remote test succeeded: $ComputerName."}
    else
        {write-host "Remote test failed: $ComputerName."}
}

if (CanPing $computername) {CanRemote $computername}

若要运行此脚本,请在脚本名称后面键入参数名称。 例如:

C:\PS> .\test-remote.ps1 -computername Server01

Ping succeeded: Server01
Remote test failed: Server01

有关 Param 语句和函数参数的详细信息,请参阅 about_Functionsabout_Functions_Advanced_Parameters

编写针对脚本的帮助

可以使用以下两种方法之一为脚本编写帮助主题:

  • 基于注释的脚本帮助

    使用注释中的特殊关键字创建帮助主题。 要为脚本创建基于评论的帮助,必须将注释放置在脚本文件的开头或末尾。 有关基于注释的帮助的详细信息,请参阅 about_Comment_Based_Help

  • 基于 XML 的脚本帮助

    创建一个基于XML的帮助主题,例如通常为CMDLET创建的类型。 如果要将帮助主题翻译成多种语言,则需要基于 XML 的帮助。

若要将脚本与基于 XML 的帮助主题相关联,请使用 .ExternalHelp 帮助注释关键字。 有关 ExternalHelp 关键字的详细信息,请参阅 about_Comment_Based_Help。 有关基于 XML 的帮助的详细信息,请参阅如何编写 Cmdlet 帮助

返回退出值

默认情况下,脚本在脚本结束时不会返回退出状态。 必须使用 exit 语句从脚本返回退出代码。 默认情况下,exit 语句会返回 0。 可以提供数字值以返回其他退出状态。 非零退出代码通常表示失败。

在 Windows 上,允许 [int]::MinValue[int]::MaxValue 之间的任意数字。

在 Unix 上,只允许 [byte]::MinValue (0) 和 [byte]::MaxValue (255) 之间的正数。 -1-255 范围内的负数将通过加上 256 自动转换为正数。例如,-2 将转换为 254

在 PowerShell 中,exit 语句设置 $LASTEXITCODE 变量的值。 在Windows Command Shell (cmd.exe) 中,Exit 语句设置了 %ERRORLEVEL% 环境变量的值。

任何非数字或平台特定范围之外的参数都将转换为 0 值。

脚本作用域和点源

每个脚本都在各自的作用域内运行。 脚本中创建的功能,变量,别名和驱动器仅存在于脚本作用域中。 无法在脚本运行的作用域中访问这些项目或它们的值。

要在不同的作用域内运行脚本,可以指定作用域,例如全局或本地,也可以点源脚本。

通过点源功能,可以在当前作用域而不是脚本作用域中运行脚本。 运行点源脚本时,脚本中的命令就像在命令提示符中键入一样运行。 脚本创建的函数、变量、别名和驱动器是在正在使用的作用域中创建的。 脚本运行后,可以使用创建的项并在会话中访问其值。

若要对脚本进行脚本点源,请在脚本路径前键入点 (.) 和空格。

例如:

. C:\scripts\UtilityFunctions.ps1

. .\UtilityFunctions.ps1

运行 UtilityFunctions.ps1 脚本后,脚本创建的函数和变量将添加到当前作用域。

例如, UtilityFunctions.ps1 脚本会创建 New-Profile 函数和 $ProfileName 变量。

#In UtilityFunctions.ps1

function New-Profile
{
  Write-Host "Running New-Profile function"
  $profileName = split-path $profile -leaf

  if (test-path $profile)
    {write-error "Profile $profileName already exists on this computer."}
  else
    {new-item -type file -path $profile -force }
}

如果在自己的脚本作用域内运行 UtilityFunctions.ps1 脚本,则 New-Profile 函数和 $ProfileName 变量仅在脚本运行时存在。 脚本退出时,将删除函数和变量,如以下示例所示。

C:\PS> .\UtilityFunctions.ps1

C:\PS> New-Profile
The term 'new-profile' is not recognized as a cmdlet, function, operable
program, or script file. Verify the term and try again.
At line:1 char:12
+ new-profile <<<<
   + CategoryInfo          : ObjectNotFound: (new-profile:String) [],
   + FullyQualifiedErrorId : CommandNotFoundException

C:\PS> $profileName
C:\PS>

点源脚本并运行该脚本时,该脚本会在作用域中的会话中创建 New-Profile 函数和 $ProfileName 变量。 脚本运行后,可以在会话中使用 New-Profile 函数,如以下示例所示。

C:\PS> . .\UtilityFunctions.ps1

C:\PS> New-Profile

    Directory: C:\Users\juneb\Documents\WindowsPowerShell

    Mode    LastWriteTime     Length Name
    ----    -------------     ------ ----
    -a---   1/14/2009 3:08 PM      0 Microsoft.PowerShellISE_profile.ps1

C:\PS> $profileName
Microsoft.PowerShellISE_profile.ps1

有关作用域的详细信息,请参阅 about_Scopes

模块中的脚本

模块是一组可以作为单元分发的相关 PowerShell 资源。 可以使用模块来组织脚本、函数和其他资源。 还可以使用模块将代码分发给其他人,以及从受信任的源获取代码。

可以在模块中包含脚本,也可以创建一个脚本模块,即一个完全或主要由脚本和支持的资源组成的模块。 脚本模块只是一个文件扩展名为 .psm1 的脚本。

若要详细了解模块,请参阅 about_Modules

其他脚本功能

PowerShell 具有许多可用于脚本的有用功能。

  • #Requires - 可以使用 #Requires 语句来防止脚本在没有指定的模块或管理单元以及指定版本的 PowerShell 的情况下运行。 有关详细信息,请参阅 about_Requires

  • $PSCommandPath - 包含正在运行的脚本的完整路径和名称。 此参数在所有脚本中均有效。 此自动变量在 PowerShell 3.0 中引入。

  • $PSScriptRoot - 包含从中运行脚本的目录。 在 PowerShell 2.0 中,此变量仅在脚本模块 (.psm1) 中有效。 从 PowerShell 3.0 开始,它在所有脚本中均有效。

  • $MyInvocation - $MyInvocation 自动变量包含有关当前脚本的信息,包括有关如何启动或“调用”的信息。可以使用此变量及其属性在运行脚本时获取有关脚本的信息。 例如,$MyInvocation.MyCommand.Path 变量包含脚本的路径和文件名。 $MyInvocation.Line 包含启动脚本的命令,包括所有参数和值。

    从 PowerShell 3.0 开始, $MyInvocation 有两个新属性,提供有关调用当前脚本的脚本的信息。 仅当调用方是脚本时,才会填充这些属性的值。

    • PSCommandPath 包含调用当前脚本的脚本的完整路径和名称。

    • PSScriptRoot 包含调用当前脚本的脚本的目录。

    与包含当前脚本信息的 $PSCommandPath$PSScriptRoot 自动变量不同,$MyInvocation 变量的 pscommandpathpsscriptroot 属性包含有关调用当前脚本的脚本的信息。

  • 数据部分 - 可以使用 Data 关键字将数据与脚本中的逻辑分开。 数据部分还可以使本地化更容易。 有关详细信息,请参阅 about_Data_Sectionsabout_Script_Internationalization

  • 脚本签名 - 可以向脚本添加数字签名。 根据执行策略,可以使用数字签名来限制可能包含不安全命令的脚本的运行。 有关详细信息,请参阅 about_Execution_Policiesabout_Signing

另请参阅