PowerShell 技能连载 - 轻松地格式化数字

用户常常需要格式化数字并且限制小数的位数,或者是在左侧补零。有一个简单标准的方法:使用操作符 -f

以下代码的作用是左侧补零:

$number = 68
'{0:d7}' -f $number

这段代码将生成一个左补零的 7 位数字。调整“d”后的数字可以控制位数。

要限制小数的位数,请使用“n”来代替“d”。这一次,“n”后的数字控制小数的位数:

$number = 35553568.67826738
'{0:n1}' -f $number

类似地,用“p”来格式化百分比:

$number = 0.32562176536
'{0:p2}' -f $number

PowerShell 技能连载 - 消除重复

Sort-Object 有一个很棒的特性:使用 -Unique 参数,您可以移除重复对象:

这也可以用于对象类型的结果。请看这个例子,它将从您的系统事件日志中获取最后的 40 条错误:

它的结果也许完全正确,但是您实际上可能得到很多重复的条目。

通过使用 -Unique 参数,您可以基于多个属性消除重复的结果:

这样,您再也看不到多于一条具有相同 InstanceID 消息的结果了。

您可以再次对结果排序,以使得结果按时间排序。

所以结论是:Sort-Objects-Unique 参数可以一次性应用到多个属性上。

PowerShell 技能连载 - 担心隐藏的输入密码请求

您可以在任何主机上运行 PowerShell,powershell.exe 和 powershell_ise.exe 随着 Windows 发布。比起简单的 PowerShell 控制台,许多人更喜欢图形化的 ISE 编辑器。

一旦您开始使用控制台程序,您必须意识到 ISE 编辑器没有真实的控制台。在 ISE 编辑器中,任何时候一个控制台程序想要与您互动时,它将会运行失败。

所以 choice.exe 在控制台中工作正常,但是您在 ISE 编辑器中运行相同的命令时,却没有办法使 choice.exe 接收您的按键。

有些时候,这可能会导致意外的结果。当您在 ISE 编辑器中运行 driverquery.exe 加上 /S Servername 参数从远程系统中读取驱动器信息时,编辑器会假死。

当您在控制台中运行相同的命令时,您会知道原因:driverquery.exe 可能会显示一个提示并期望输入密码。ISE 编辑器无法处理这个提示和您的输入——由于它没有控制台缓冲区。

所以当您的脚本用到了控制台应用程序时,您最好在传统的 PowerShell 控制台中运行它们。

PowerShell 技能连载 - 为对象增加信息

有时会遇到这样的需求:需要向命令的执行结果增加额外的信息。也许您从不同的机器获取信息,并且希望保存数据来源的引用。或者,您也许希望增加一个日期,以便知道数据是何时创建的。

向对象附加信息(增加额外的信息列)是很简单的。这段代码将为一个服务列表增加新的“SourceComputer”属性以及日期。

Get-Service |
  Add-Member -MemberType NoteProperty -Name SourceComputer -Value $env:COMPUTERNAME -PassThru |
  Add-Member -MemberType NoteProperty -Name Date -Value (Get-Date) -PassThru |
  Select-Object -Property Name, Status, SourceComputer, Date

请记着您新增的属性需要在使用 Select-Object 以及显式地要求显式它们的时候,才会在结果中显示出来。

PowerShell 技能连载 - 用 Select-Object -First 节省时间!

Select-Object 命令有一个 -First 参数,接受一个数字值。它将只返回前 x 个元素。听起来挺简单的,而且它的确就这么简单。

以下代码从您的 Windows 文件夹中获取前 4 个 PowerShell 脚本:

从 PowerShell 3.0 开始,-First 不仅选择指定数量的结果,而且它还通知管道的上游命令,告知它操作已完成,有效地中止管道操作。

所以如果您想得到某个命令一定数量的结果,那么您总是可以使用 Select-Object -First x ——在特定的场景里这可以显著地加速您的代码执行效率。

我们假设您需要在您的用户文件夹之下的某个地方找一个名为“test.txt”的文件,并且假设只有一个这个名字的文件。而您只是不知道它放在哪个位置,那么您可以使用 Get-ChildItem-Recurse 来递归查找所有的文件夹:

Get-ChildItem -Path $home -Filter test.txt -Recurse -ErrorAction SilentlyContinue

当您执行这段代码时,Get-ChildItem 最终将找到您的文件——并且继续搜索您的文件夹树,也许要持续几分钟才能找完。因为它不知道是否有其它的文件。

所以,您知道的,如果您事先确定结果的数量,那么试试以下的代码:

Get-ChildItem -Path $home -Filter test.txt -Recurse -ErrorAction SilentlyContinue |
  Select-Object -First 1

这一次,Get-ChildItem 将会在找到一个文件后立即停止。

PowerShell 技能连载 - 展开字符串中的变量

要在一个字符串中插入一个变量,您也许已经知道可以使用如下的双引号方式:

$domain = $env:USERDOMAIN
$username = $env:USERNAME

"$domain\$username"

对于 PowerShell 来说这些变量的起止范围是没有歧义的。所以它可以工作正常。然而试试以下代码:

$domain = $env:USERDOMAIN
$username = $env:USERNAME

"$username: located in domain $domain"

这段代码执行失败了,这是因为 PowerShell 在变量中添加了冒号(从语法彩色中也可以看出)。

您可以采用 PowerShell 的反引号来为特殊字符(比如说冒号)转义:

$domain = $env:USERDOMAIN
$username = $env:USERNAME

"$username`: located in domain $domain"

如果问题不是由于特殊字符引起的,那么这种方法没有作用:

"Current Background Color: $host.UI.RawUI.BackgroundColor"

语法高亮提示双引号引起来的字符串中只解析出了变量,而其它部分(变量名之后的部分,比如说对象的属性)并没有解析出来。

要解决这个问题,您需要使用以下的方法之一:

"Current Background Color: $($host.UI.RawUI.BackgroundColor)"
'Current Background Color: ' + $host.UI.RawUI.BackgroundColor
'Current Background Color: {0}' -f $host.UI.RawUI.BackgroundColor

PowerShell 技能连载 - 使用别名来启动 Windows 组件

PowerShell 不仅是一个自动化操作的语言,而且是另一个用户操作界面。如果您不喜欢图形界面,那么练习使用 PowerShell 通过简单的别名来启动您需要的工具。

例如,要打开设备管理器,您可以使用它的原始名称:

如果您不想记忆这个名称,那么使用别名:

如您所见,要打开设备管理器,您现在所要做的只是键入“DeviceManager”。您也可以只键入“Device”然后按下 TAB 键来使用自动完成功能。

当关闭 PowerShell 时,定义的别名将会消失。所以要保持您定义的别名有效,请将 Set-Alias 命令加入您的配置脚本。配置脚本的路径可以在 $profile 中找到。如果这个目录不存在,您可能需要事先创建这个文件(以及它所在的文件夹)。Test-Path 可以检测它是否已经存在。

PowerShell 技能连载 - 过滤命令输出的文本

比较操作符作用于数组时,它们的作用和过滤器相似。所以许多输出多行文本的控制台命令可以使用比较操作符。

以下例子将使用 netstat.exe 来获取已连接上的网络连接,然后过滤出连到名字包含“stor”的服务器的连接,最后用 ipconfig 来获取当前的 IPv4 地址:

这个技巧是将控制台命令用 @() 括起来,确保结果为一个数组。

PowerShell 技能连载 - 持有一个进程的句柄

当您打开一个 EXE 文件时,PowerShell 将会开心地启动它,然后什么也不管:

如果您希望持有该进程的句柄,比如希望获得它的进程 ID,或者过一会儿检查该进程运行得如何,或者要中止它,请使用 Start-Process–PassThru 参数。以下代码将返回一个进程对象:

PowerShell 技能连载 - 使用 $PSScriptRoot 加载资源

从 PowerShell 3.0 开始,有一个称为 $PSScriptRoot 的变量可用。该变量之前只在模块中可用。它总是指向当前脚本所在的文件夹(所以它仅在您明确地保存了它以后才生效)。

您可以使用 $PSScriptRoot 来加载相对于您脚本位置的额外资源。例如,如果您打算将一些函数放在同一个文件夹中的一个独立的“library”脚本中,以下代码将加载该 library 脚本并且导入它包含的所有函数:

# this loads the script "library1.ps1" if it is located in the very
# same folder as this script.
# Requires PowerShell 3.0 or better.

. "$PSScriptRoot\library1.ps1"

类似地,如果您希望将您的 library 脚本保存在一个子文件夹中,请试验以下脚本(假设库脚本放在您脚本所在文件夹中的“resources”子文件夹下):

# this loads the script "library1.ps1" if it is located in the subfolder
# "resources" in the folder this script is in.
# Requires PowerShell 3.0 or better.

. "$PSScriptRoot\resources\library1.ps1"