我正在编写一个 PowerShell 脚本来检查服务器的某些属性。当我执行以下 cmdlet 时为了,输出没有按照我期望的顺序显示。
Write-Host "=== Operating System ==="
Get-WmiObject -computername $ServerName -class Win32_OperatingSystem | select Caption | ft -HideTableHeaders -AutoSize
Write-Host "=== Pagefile ==="
Get-WmiObject -ComputerName $ServerName -Class Win32_PageFileSetting
Write-Host "=== Locale ==="
Get-WmiObject -ComputerName $ServerName -class Win32_OperatingSystem | select Locale | ft -HideTableHeaders -AutoSize | Write-Output
输出:
=== 操作系统 ===
Microsoft Windows Server 2012 标准版
=== 页面文件 ===
=== 区域设置 ===
最大大小 名称 标题
----------- ---- -------
4000 C:\pagefile.sys C:\ 'pagefile.sys'0813
为什么页面文件输出没有显示在标题下?在我看来,这Get-WmiObject
是异步运行并立即写入下一个标题(区域设置)。
我该如何避免这种情况?使用 Write-Output 时可以正常工作,但无法使用彩色文本。
答案1
屏幕上的某些文本来自经过PowerShell 管道,有些来自于Write-Host
不接触管道而立即打印到屏幕上。
收集有关页面文件的 WMI 信息需要时间并输出到管道,但脚本在管道完成之前就已经开始打印区域设置消息。
对于 (操作系统) 来说,这种情况不会发生,因为该管道以ft -AutoSize
(Format-Table) 结束,这会暂停脚本并等待管道中的所有内容,以便在决定如何调整列大小时拥有尽可能多的信息。一旦ft
打印到屏幕上,脚本就会移动到 PageFile 消息。
您应该使用Write-Output
通过管道发送文本,这样它可以很好地与其他 PowerShell 功能(如重定向、在文件和变量中收集命令输出)集成。
Jeffrey Snover(PowerShell 首席架构师)在他的博客中写了为什么Write-Host
不推荐使用:http://www.jsnover.com/blog/2013/12/07/write-host-considered-harmful/
以下是描述行为的链接Format-Table
:
优化列宽
由于管道实时处理结果,PowerShell 无法知道列元素将占用多大的空间。因此,它在调整列大小时往往会比较宽。如果指定 -AutoSize 参数,Format-Table 将首先收集所有结果,然后再设置所有元素的最大宽度。您可以优化输出,但结果将不再实时输出: