我尽可能使用PowerShell来快速轻松地编写脚本;在我的工作中很多时候我会用它来进行数据解析,日志文件筛选或创建CSV \ Text文件。
我无法弄清楚的一点是,执行某些数据\ IO任务的效率非常低。我认为这与它处理管道的方式有关,它处理管道的方式或者我还没有理解的东西。
如果您采用以下逻辑生成ABC123 ID,请在PowerShell中编译并执行它,完成时间不到1分钟:
$source = @'
public static System.Collections.Generic.List<String> GetIds()
{
System.Collections.Generic.List<String> retValue = new System.Collections.Generic.List<String>();
for (int left = 97; left < 123; left++)
{
for (int middle = 97; middle < 123; middle++)
{
for (int right = 97; right < 123; right++)
{
for (int i = 1; i < 1000; i++)
{
String tmp = String.Format("{0}{1}{2}000", (char)left, (char)middle, (char)right);
retValue.Add(String.Format("{0}{1}", tmp.Substring(0, tmp.Length - i.ToString().Length), i));
}
}
}
}
return retValue;
}
'@
$util = Add-Type -Name "Utils" -MemberDefinition $source -PassThru -Language CSharp
$start = get-date
$ret = $util::GetIds()
Write-Host ("Time: {0} minutes" -f ((get-date)-$start).TotalMinutes)
现在采用相同的逻辑,通过PowerShell运行它而不作为程序集进行编译,并且需要小时才能完成
$start = Get-Date
$retValue = @()
for ($left = 97; $left -lt 123; $left++)
{
for ($middle = 97; $middle -lt 123; $middle++)
{
for ($right = 97; $right -lt 123; $right++)
{
for ($i = 1; $i -lt 1000; $i++)
{
$tmp = ("{0}{1}{2}000" -f [char]$left, [char]$middle, [char]$right)
$retValue += ("{0}{1}" -f $tmp.Substring(0, $tmp.Length - $i.ToString().Length), $i)
}
}
}
}
Write-Host ("Time: {0} minutes" -f ((get-date)-$start).TotalMinutes)
为什么?是否存在某种过度的类型转换或我使用的低效操作会降低性能?
答案 0 :(得分:4)
你在这里扼杀你的表现:
$retValue += ("{0}{1}" -f $tmp.Substring(0, $tmp.Length - $i.ToString().Length), $i)
阵列添加是非常昂贵的#34;操作。你正在做的基本上是每次创建一个全新的数组,由原始数组和新元素组成。
编辑:这种阵列添加不仅效率低下,而且完全没必要。您所要做的就是将这些值输出到管道,并将结果分配回变量。
$start = Get-Date
$retValue =
for ($left = 97; $left -lt 123; $left++)
{
for ($middle = 97; $middle -lt 123; $middle++)
{
for ($right = 97; $right -lt 123; $right++)
{
for ($i = 1; $i -lt 1000; $i++)
{
$tmp = ("{0}{1}{2}000" -f [char]$left, [char]$middle, [char]$right)
"{0}{1}" -f $tmp.Substring(0, $tmp.Length - $i.ToString().Length), $i
}
}
}
}
Write-Host ("Time: {0} minutes" -f ((get-date)-$start).TotalMinutes)
Time: 1.866812045 minutes