Как передать переменную по ссылке на задание или пространство выполнения Powershell?
У меня есть работа в Powershell.
$cmd = {
param($a, $b)
$a++
$b++
}
$a = 1
$b = 2
Start-Job -ScriptBlock $cmd -ArgumentList $a, $b
Как пройти $a
а также $b
по ссылке, поэтому, когда работа будет выполнена, они будут обновлены? В качестве альтернативы, как передать переменные по ссылке на пространства выполнения?
3 ответа
Простой пример, который я только что написал (не обращайте внимания на грязный код)
# Test scriptblock
$Scriptblock = {
param([ref]$a,[ref]$b)
$a.Value = $a.Value + 1
$b.Value = $b.Value + 1
}
$testValue1 = 20 # set initial value
$testValue2 = 30 # set initial value
# Create the runspace
$Runspace = [runspacefactory]::CreateRunspace()
$Runspace.ApartmentState = [System.Threading.ApartmentState]::STA
$Runspace.Open()
# create the PS session and assign the runspace
$PS = [powershell]::Create()
$PS.Runspace = $Runspace
# add the scriptblock and add the argument as reference variables
$PS.AddScript($Scriptblock)
$PS.AddArgument([ref]$testValue1)
$PS.AddArgument([ref]$testValue2)
# Invoke the scriptblock
$PS.BeginInvoke()
После выполнения этого значения тестов обновляются, так как они передаются с помощью ref.
Передача параметров по ссылке всегда неудобна в PowerShell и, вероятно, не будет работать для заданий PowerShell в любом случае, как отметил @bluuf.
Я бы, вероятно, сделал что-то вроде этого:
$cmd = {
Param($x, $y)
$x+1
$y+1
}
$a = 1
$b = 2
$a, $b = Start-Job -ScriptBlock $cmd -ArgumentList $a, $b |
Wait-Job |
Receive-Job
Приведенный выше код передает переменные $a
а также $b
к блоку сценария и назначает измененные значения обратно переменным после получения выходных данных задания.
более полный сценарий с примером.
- это также должно включать способность проходить
$host
или что-то, чтобы сделатьwrite-host
из переданного скрипта вывод в консоль. но у меня нет времени разбираться, как это сделать.
$v = 1
function newThread ([scriptblock]$script, [Parameter(ValueFromPipeline)]$param, [Parameter(ValueFromRemainingArguments)]$args) {
process {
$Powershell = [powershell]::Create()
$Runspace = [runspacefactory]::CreateRunspace()
# allows to limit commands available there
# $InitialSessionState = InitialSessionState::Create()
# $Runspace.InitialSessionState = $InitialSessionState
$Powershell.Runspace = $Runspace
$null = $Powershell.AddScript($script)
$null = $Powershell.AddArgument($param)
foreach ($v_f in $args) {
$null = $Powershell.AddArgument($v_f)
}
$Runspace.Open()
$Job = $Powershell.BeginInvoke()
[PSCustomObject]@{
Job=$Job
Powershell=$Powershell
}
}
}
$script = {
param([ref]$v,$v2)
$v.Value++
$v2
}
$thread = newThread $script ([ref]$v) 3
do {} until ($thread.Job.IsCompleted)
$v1 = $thread.Powershell.EndInvoke($thread.Job)
$thread.Powershell.Dispose()
write-host "end $($v,$v1[0])"