将“本机”对象传递给后台作业

时间:2013-03-13 10:29:37

标签: powershell scope psobject

这是我想以某种方式实现的目标。

我有一个定义一些对象的自定义程序集。在我的脚本中,我创建了一个自定义对象,我想将其传递给脚本块,保持该对象的行为。

Add-Type -AssemblyName MyCustomDLL

$global:object = new-object MyCustomDLL.MyCustomObject()
$object | gm

$jobWork = { param ($object) $object | gm } # I'd like to keep my object behavior in that block

$job = Start-Job -ScriptBlock $jobWork -ArgumentList $object
Wait-Job $job
Receive-Job $job

我该怎么做或达到同样的效果?谢谢你的帮助

2 个答案:

答案 0 :(得分:9)

您可以将PowerShellBeginInvokeEndInvoke一起使用,而不是后台作业。以下是在“作业”中传递活动对象的简单但有效的示例,在那里更改它,获得结果:

# live object to be passed in a job and changed there
$liveObject = @{ data = 42}

# job script
$script = {
    param($p1)
    $p1.data # some output (42)
    $p1.data = 3.14 # change the live object data
}

# create and start the job
$p = [PowerShell]::Create()
$null = $p.AddScript($script).AddArgument($liveObject)
$job = $p.BeginInvoke()

# wait for it to complete
$done = $job.AsyncWaitHandle.WaitOne()

# get the output, this line prints 42
$p.EndInvoke($job)

# show the changed live object (data = 3.14)
$liveObject

答案 1 :(得分:4)

后台作业建立在PowerShell远程处理之上,因此,在传递对象时执行类似的操作。他们会对它们进行序列化/反序列化,而不是传递它们的复杂性。

我的猜测是,获取复杂对象的唯一方法就是将构造函数参数和/或操作作为-ArgumentList传递,并在作业中创建对象

在这种情况下,添加组件也必须是工作的一部分:

Start-Job {
    param ($ConstructorArguments)
    Add-Type -AssemblyName MyCustomDll
    $object = New-Object MyCustomDll.MyCustomObject $ConstructorArguments
    $object | Get-Member
} -ArgumentList Foo, Bar | Wait-Job | Receive-Job