Powershell脚本以交互方式工作,而不是在脚本中

时间:2013-08-09 02:34:02

标签: powershell powershell-v2.0 powershell-v3.0

代码以交互方式工作,在PS1文件中它没有。重现,打开powershell,粘贴该函数,然后运行get-job来查看任务。输入get-job |完成后删除作业,然后将代码放入PS1文件中,它只运行前两个,然后退出。

function RunJobFromQueue
{

    if( $queue.Count -gt 0)

    {
        $cn = $queue.Dequeue()

        $j = Start-Job -name $cn -ScriptBlock {param($x); Start-Sleep -Seconds 10;"output - " + $x} -ArgumentList $cn

        Register-ObjectEvent -InputObject $j -EventName StateChanged -Action {RunJobFromQueue; Unregister-Event $eventsubscriber.SourceIdentifier; Remove-Job $eventsubscriber.SourceIdentifier } | Out-Null
    }

}

$maxConcurrentJobs = 2
$jobInput = "test1", "test2", "test3", "test4", "test5", "test6"
$queue = [System.Collections.Queue]::Synchronized( (New-Object System.Collections.Queue) )
foreach($item in $jobInput) {$queue.Enqueue($item)}

for( $i = 0; $i -lt $maxConcurrentJobs; $i++){RunJobFromQueue}

1 个答案:

答案 0 :(得分:4)

在我看来:它在脚本中失败了,因为:

  • 启动脚本,在脚本范围中定义函数和变量
  • 脚本在定义2个初始作业后退出
  • 作业完成,已注册的操作已运行,但您的功能对其不可见(在脚本范围中定义),因此失败。

解决/解决它的两个选项:

  • 在全局范围内定义RunJobFromQueue函数(function Global:RunJobFromQueue$global:queue
  • dot-source a script(. .\YourScript.ps1

这可以很好地交互式工作,因为在这种情况下你可以在全局范围内定义函数/变量,所以-Action可以很好地找到它们。