PowerShell脚本效率

时间:2015-03-05 18:07:42

标签: c# performance powershell

我尽可能使用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)

为什么?是否存在某种过度的类型转换或我使用的低效操作会降低性能?

1 个答案:

答案 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