start-job并行运行命令,并在结束时输出结果

时间:2013-03-27 23:25:57

标签: powershell parallel-processing start-job

我试图在服务器列表中获取特定的KBXXXXXX,但是一旦我的脚本一个服务器,它需要时间并返回结果并返回然后转到下一个。这个脚本对我来说非常好。 我希望我的脚本启动并获取修补程序作为工作和其他过程只是为了收集结果并显示它们。

$servers = gc .\list.txt 
foreach ($server in $servers) 
{ 
    $isPatched = (Get-HotFix -ComputerName $server | where HotFixID -eq 'KBxxxxxxx') -ne $null 
    If ($isPatched) 
    { 
    write-host $server + "Exist">> .\patchlist.txt} 
    Else  
    { 
    Write-host $server +"Missing"
$server  >> C:\output.txt
    } 

}

使列表执行得更快而不是串行运行的目标。

3 个答案:

答案 0 :(得分:3)

使用Powershell V2,您可以使用@Andy答案中的作业,也可以使用此链接中的更详细信息Can Powershell Run Commands in Parallel?

使用PowerShell V2,您可能还想使用runspaces

查看此脚本http://gallery.technet.microsoft.com/scriptcenter/Foreach-Parallel-Parallel-a8f3d22b

使用PowerShell V3,您可以使用foreach -parallel选项。

例如

(NB Measure-Command只是用于计时,因此您可以进行比较)

Workflow Test-My-WF {  
  param([string[]]$servers)

  foreach -parallel ($server in $servers) {

    $isPatched = (Get-HotFix -ComputerName $server | where {$_.HotFixID -eq 'KB9s82018'}) -ne $null     
    If ($isPatched) 
    { 
        $server | Out-File -FilePath "c:\temp\_patchlist.txt" -Append
    } 
    Else  
    { 
        $server | Out-File -FilePath "c:\temp\_output.txt" -Append 
    } 
  }
}

Measure-Command -Expression { Test-My-WF   $servers }

答案 1 :(得分:0)

为此使用PowerShell作业。

小命令:

  • 获取-作业
  • 接收-作业
  • 移除-作业
  • 启动工作
  • 停止-作业
  • 等待-作业

这是一个未经测试的例子:

$check_hotfix = {
    param ($server)
    $is_patched = (Get-HotFix -ID 'KBxxxxxxx' -ComputerName $server) -ne $null 
    if ($is_patched) { 
        Write-Output ($server + " Exist") 
    } else { 
        Write-Output ($server + " Missing")
    }
}

foreach ($server in $servers) {
    Start-Job -ScriptBlock $check_hotfix -ArgumentList $server | Out-Null
}

Get-Job | Wait-Job | Receive-Job | Set-Content patchlist.txt

答案 2 :(得分:0)

使用查询cmdlet中内置的多台计算机的能力,而不是使用作业。 Microsoft的许多cmdlet(尤其是用于系统管理的cmdlet)将一个字符串数组作为-Computername参数的输入。传入服务器列表,cmdlet将查询所有服务器。大多数具有此功能的cmdlet将以串行方式查询服务器,但Invoke-Command将并行执行。

我没有对此进行测试,因为我目前没有启动Windows,但这应该让你开始(按顺序)。

$servers = gc .\list.txt 
$patchedServers = Get-HotFix -ComputerName $servers | where HotFixID -eq 'KBxxxxxxx'|select machinename

$unpatchedServers = compare-object -referenceobject $patchedServers -differenceobject $servers -PassThru

$unpatchedServers |out-file c:\missing.txt;
$patchedServers|out-file c:\patched.txt;

同时:

$servers = gc .\list.txt 
$patchedServers = invoke-command -computername $servers -scriptblock {Get-HotFix | where HotFixID -eq 'KBxxxxxxx'}|select -expandproperty pscomputername |sort -unique

和以前一样,我目前还没有适合的Windows版本来测试上述&检查输出,但这是一个起点。