在值的属性上对PowerShell哈希表进行排序

时间:2013-08-01 17:15:34

标签: sorting powershell hashtable

我在排序哈希表时遇到问题。我已经将我的代码分解为只是必需品,以免用原始脚本压倒任何人。

Write-Host "PowerShell Version = " ([string]$psversiontable.psversion) 
$h = @{}
$Value = @{SortOrder=1;v1=1;}
$h.Add(1, $Value)
$Value = @{SortOrder=2;v1=1;}
$h.Add(2, $Value)
$Value = @{SortOrder=3;v1=1;}
$h.Add(3, $Value)
$Value = @{SortOrder=4;v1=1;}
$h.Add(4, $Value)

Write-Host "Ascending"
foreach($f in $h.GetEnumerator() | Sort-Object Value.SortOrder)
{
    Write-Host $f.Value.SortOrder
}

Write-Host "Descending"
foreach($f in $h.GetEnumerator() | Sort-Object Value.SortOrder -descending)
{
    Write-Host $f.Value.SortOrder
}

输出

PowerShell Version =  3.0
Ascending
2
1
4
3
Descending
2
1
4
3

我确信这只是一个不知道Sort-Object正确用法的简单案例。排序在Sort-Object Name上正常运行,所以可能与不知道如何处理Value.SortOrder有关?

2 个答案:

答案 0 :(得分:23)

Sort-Object接受用于排序的属性名称或脚本块。由于您尝试对属性的属性进行排序,因此您需要使用脚本块:

Write-Host "Ascending"
$h.GetEnumerator() | 
    Sort-Object { $_.Value.SortOrder } | 
    ForEach-Object {  Write-Host $_.Value.SortOrder }

Write-Host "Descending"
$h.GetEnumerator() |
    Sort-Object { $_.Value.SortOrder } -Descending |
    ForEach-Object { Write-Host $_.Value.SortOrder }

您可以使用Where-Object cmdlet进行过滤:

Write-Host "Ascending"
$h.GetEnumerator() | 
    Where-Object { $_.Name -ge 2 } |
    Sort-Object { $_.Value.SortOrder } | 
    ForEach-Object {  Write-Host $_.Value.SortOrder }

您通常希望在任何Where-Object cmdlet之前放置Sort-Object ,因为它会使排序更快。

答案 1 :(得分:2)

我使用哈希表作为频率表来计算文件名中单词的出现次数。

$words = @{}
get-childitem *.pdf | foreach-object -process {
    $name = $_.name.substring($_.name.indexof("-") + 1, $_.name.indexof(".") - $_.name.indexof("-") - 1)
    $name = $name.replace("_", " ")
    $word = $name.split(" ")[0]
    if ( $words.contains($word) ){
        $words[$word] = $words[$word] + 1 
    }else{
        $words.add($word, 1)
    }
}
$words.getenumerator() | sort-object -property value

这是魔术的最后一行,在值(频率)上对哈希表进行排序。