如何在同一个Runspace中并行运行两个cmdlet。我正在使用C#。
InitialSessionState iss = InitialSessionState.CreateDefault();
iss.AuthorizationManager = new AuthorizationManager("MyShellId");
iss.ImportPSModule(new string[] { "MSOnline" });
Runspace powerShellRunspace = RunspaceFactory.CreateRunspace(iss);
在两个线程中,我将使用相同的运行空间运行cmdlet。
Pipeline pipeLine = powerShellRunspace.CreatePipeline();
pipeLine.Commands.Add(shellCommand);
pipeLine.Input.Close();
pipeLine.Invoke();
pipeLine.Output.DataReady += new EventHandler(processData); //processData is a method which processes data emitted by pipeline as and when it comes to avoid out of memory
if (pipeLine.Error != null && pipeLine.Error.Count > 0) {
Collection<Object> errors = (Collection<Object>)(pipeLine.Error.ReadToEnd());
//process those errors
}
但是当两个线程同时使用相同的运行空间来运行cmdlet时。我正在异常,“管道没有执行,因为管道已经在执行。管道不能同时执行。”
出于性能原因,我需要使用相同的运行空间。如何实现我的目标?
答案 0 :(得分:2)
你看过System.Management.Automation.Runspaces.RunspacePool班了吗?使用它和InitialSessionState可以帮助消除模块导入的开销,因为每个池只执行一次而不是每个运行空间。如果你正在寻找powershell命令的异步执行,这里是一个非常基本的,而不是生产就绪的例子:(注意我不是在使用Visual Studio的计算机上,但这应该是正确的)
InitialSessionState iss = InitialSessionState.CreateDefault();
iss.AuthorizationManager = new AuthorizationManager("MyShellId");
iss.ImportPSModule(new string[] { "MSOnline" });
#set commands we want to run concurrently
string[] commands = new string[4] {
"Start-Sleep -Seconds 5; 'Hi from #1'",
"Start-Sleep -Seconds 7; 'Hi from #2'",
"Start-Sleep -Seconds 3; 'Hi from #3'",
"throw 'Danger Will Robinson'"
};
Dictionary<PowerShell, IAsyncResult> dict = new Dictionary<PowerShell, IAsyncResult>();
//this loads the InitialStateSession for all instances
//Note you can set the minimum and maximum number of runspaces as well
using(RunspacePool rsp = RunspaceFactory.CreateRunspacePool(iss))
{
rsp.SetMinRunspaces(5);
rsp.SetMaxRunspaces(10);
rsp.Open();
foreach(string cmd in commands)
{
PowerShell ps = PowerShell.Create();
ps.AddScript(cmd);
ps.RunspacePool = rsp;
//Add parameters if needed with ps.AddParameter or ps.AddArgument
dict.Add(ps,ps.BeginInvoke());
}
do{
List<PowerShell> toBeRemoved = new List<PowerShell>();
foreach(KeyValuePair<PowerShell, IAsyncResult> kvp in dict)
{
if(kvp.Value.IsCompleted)
{
try
{
PSDataCollection<PSObject> objs = kvp.Key.EndInvoke(kvp.Value);
foreach(PSObject obj in objs)
{
Console.WriteLine(obj);
}
}
catch (Exception e)
{
Console.WriteLine(e);
}
finally
{
toBeRemoved.Add(kvp.Key);
}
}
}
foreach(PowerShell item in toBeRemoved)
{
dict.Remove(item);
}
//Wait before we check again
Thread.Sleep(200);
} while (dict.Count > 0)
rsp.Close();
}
//Added to keep console open
Console.Read();
这应该给出:
Hi from #3
Hi from #1
Hi from #2