使用 PowerShell 列出已安装的软件:一个免费工具

如果您是系统管理员,其中之一的工作就是在许多系统上安装、升级和删除软件。如果我告诉您,您不必再手动连接到每台机器并检查已安装的软件了,您会怎么想?实际上,您可以使用 PowerShell 列出已安装的软件!

在本博客文章中,我将向您展示如何使用 PowerShell 在本地机器和多台计算机上同时列出已安装的软件。

Listing installed software with PowerShell script

请注意,一些文章可能会告诉您执行类似 Get-WmiObject -Class win32_product 的操作。不要这样做。了解原因请看这里

已安装的软件和注册表

供参考,已安装的软件存在于三个位置:

  • 32位系统卸载注册表键
  • 64位系统卸载注册表键
  • 每个用户配置文件的卸载注册表键。

每个软件条目通常由软件的全局唯一标识符(GUID)定义。GUID 键的内部包含有关该特定软件的所有信息。要获取完整列表,PowerShell 必须枚举每个键,读取每个注册表值并解析结果。

由于正确解析这些值的代码远远超过一篇文章的容量,我预先构建了一个名为Get-InstalledSoftware的函数,使用PowerShell列出安装的软件。如下所示,该函数将所有代码封装起来,以便为您列出计算机上安装的程序。

function Get-InstalledSoftware {
    <#
	.SYNOPSIS
		检索安装在 Windows 计算机上的所有软件列表。
	.EXAMPLE
		PS> Get-InstalledSoftware
		
		此示例检索本地计算机上安装的所有软件。
	.PARAMETER ComputerName
		如果查询远程计算机,请在此处使用计算机名称。
	
	.PARAMETER Name
		您想要将查询限制为的软件标题。
	
	.PARAMETER Guid
		您想要将查询限制为的软件 GUID。
	#>
    [CmdletBinding()]
    param (
		
        [Parameter()]
        [ValidateNotNullOrEmpty()]
        [string]$ComputerName = $env:COMPUTERNAME,
		
        [Parameter()]
        [ValidateNotNullOrEmpty()]
        [string]$Name,
		
        [Parameter()]
        [guid]$Guid
    )
    process {
        try {
            $scriptBlock = {
                $args[0].GetEnumerator() | ForEach-Object { New-Variable -Name $_.Key -Value $_.Value }
				
                $UninstallKeys = @(
                    "HKLM:\Software\Microsoft\Windows\CurrentVersion\Uninstall",
                    "HKLM:\SOFTWARE\Wow6432Node\Microsoft\Windows\CurrentVersion\Uninstall"
                )
                New-PSDrive -Name HKU -PSProvider Registry -Root Registry::HKEY_USERS | Out-Null
                $UninstallKeys += Get-ChildItem HKU: | where { $_.Name -match 'S-\d-\d+-(\d+-){1,14}\d+$' } | foreach {
                    "HKU:\$($_.PSChildName)\Software\Microsoft\Windows\CurrentVersion\Uninstall"
                }
                if (-not $UninstallKeys) {
                    Write-Warning -Message 'No software registry keys found'
                } else {
                    foreach ($UninstallKey in $UninstallKeys) {
                        $friendlyNames = @{
                            'DisplayName'    = 'Name'
                            'DisplayVersion' = 'Version'
                        }
                        Write-Verbose -Message "Checking uninstall key [$($UninstallKey)]"
                        if ($Name) {
                            $WhereBlock = { $_.GetValue('DisplayName') -like "$Name*" }
                        } elseif ($GUID) {
                            $WhereBlock = { $_.PsChildName -eq $Guid.Guid }
                        } else {
                            $WhereBlock = { $_.GetValue('DisplayName') }
                        }
                        $SwKeys = Get-ChildItem -Path $UninstallKey -ErrorAction SilentlyContinue | Where-Object $WhereBlock
                        if (-not $SwKeys) {
                            Write-Verbose -Message "No software keys in uninstall key $UninstallKey"
                        } else {
                            foreach ($SwKey in $SwKeys) {
                                $output = @{ }
                                foreach ($ValName in $SwKey.GetValueNames()) {
                                    if ($ValName -ne 'Version') {
                                        $output.InstallLocation = ''
                                        if ($ValName -eq 'InstallLocation' -and 
                                            ($SwKey.GetValue($ValName)) -and 
                                            (@('C:', 'C:\Windows', 'C:\Windows\System32', 'C:\Windows\SysWOW64') -notcontains $SwKey.GetValue($ValName).TrimEnd('\'))) {
                                            $output.InstallLocation = $SwKey.GetValue($ValName).TrimEnd('\')
                                        }
                                        [string]$ValData = $SwKey.GetValue($ValName)
                                        if ($friendlyNames[$ValName]) {
                                            $output[$friendlyNames[$ValName]] = $ValData.Trim() ## 一些注册表值具有尾随空格。
                                        } else {
                                            $output[$ValName] = $ValData.Trim() ## 一些注册表值尾随空格。
                                        }
                                    }
                                }
                                $output.GUID = ''
                                if ($SwKey.PSChildName -match '\b[A-F0-9]{8}(?:-[A-F0-9]{4}){3}-[A-F0-9]{12}\b') {
                                    $output.GUID = $SwKey.PSChildName
                                }
                                New-Object -TypeName PSObject -Prop $output
                            }
                        }
                    }
                }
            }
			
            if ($ComputerName -eq $env:COMPUTERNAME) {
                & $scriptBlock $PSBoundParameters
            } else {
                Invoke-Command -ComputerName $ComputerName -ScriptBlock $scriptBlock -ArgumentList $PSBoundParameters
            }
        } catch {
            Write-Error -Message "Error: $($_.Exception.Message) - Line Number: $($_.InvocationInfo.ScriptLineNumber)"
        }
    }
}

将此函数复制并粘贴到您的 PowerShell 控制台中或将其添加到您的脚本中后,您可以使用ComputerName参数和特定的计算机名称调用它。

使用 PowerShell 列出已安装的软件

PS> Get-InstalledSoftware -ComputerName XXXXX

这样做时,您将针对安装的每个软件获取一个对象。您可以获取关于已安装的任何软件的丰富信息。

如果您提前知道软件名称,还可以使用Name参数来限制只匹配该值的软件。

例如,也许您只想检查 Microsoft Visual C++ 2005 Redistributable (x64) 是否已安装。您只需将其作为Name参数值,如下所示。

PS> Get-InstalledSoftware -ComputerName MYCOMPUTER -Name 'Microsoft VisualC++ 2005 Redistributable (x64)'

总结

使用 PowerShell 获取已安装的软件,您可以构建一个完全免费的工具,您和您的团队可以同时轻松地找到许多 Windows 计算机上安装的软件!

Source:
https://adamtheautomator.com/powershell-list-installed-software/