web-dev-qa-db-fra.com

Hashtables et ordre des clés

Est-il possible de conserver l'ordre des clés dans une table de hachage au fur et à mesure de leur ajout? Comme un mécanisme push/pop.

Exemple:

$hashtable = @{}

$hashtable.Add("Switzerland", "Bern")
$hashtable.Add("Spain", "Madrid")
$hashtable.Add("Italy", "Rome")
$hashtable.Add("Germany", "Berlin")
$hashtable

Je souhaite conserver l'ordre dans lequel j'ai ajouté les éléments à la table de hachage.

26
Phil

Il n’existe pas de solution intégrée dans PowerShell V1/V2. Vous voudrez utiliser .NET System.Collections.Specialized.OrderedDictionary :

$order = New-Object System.Collections.Specialized.OrderedDictionary
$order.Add("Switzerland", "Bern")
$order.Add("Spain", "Madrid")
$order.Add("Italy", "Rome")
$order.Add("Germany", "Berlin")


PS> $order

Name                           Value
----                           -----
Switzerland                    Bern
Spain                          Madrid
Italy                          Rome
Germany                        Berlin

Dans PowerShell V3, vous pouvez transtyper vers [ordonné]:

PS> [ordered]@{"Switzerland"="Bern"; "Spain"="Madrid"; "Italy"="Rome"; "Germany"="Berlin"}

Name                           Value
----                           -----
Switzerland                    Bern
Spain                          Madrid
Italy                          Rome
Germany                        Berlin
41
Loïc MICHEL

Vous pouvez utiliser un dictionnaire commandé à la place:

Comme ça:

$list = New-Object System.Collections.Specialized.OrderedDictionary
$list.Add("Switzerland", "Bern")
$list.Add("Spain", "Madrid")
$list.Add("Italy", "Rome")
$list.Add("Germany", "Berlin")
$list
7
Frode F.

Vous pouvez donner une clé séquentielle lorsque vous ajoutez des éléments:

$hashtable = @{}
$hashtable[$hashtable.count] = @("Switzerland", "Bern")
$hashtable[$hashtable.count] = @("Spain", "Madrid")
$hashtable[$hashtable.count] = @("Italy", "Rome")
$hashtable[$hashtable.count] = @("Germany", "Berlin")
$hashtable

Ensuite, vous pouvez obtenir des éléments triés par la clé:

echo "`nHashtable keeping the order as they were added"
foreach($item in $hashtable.getEnumerator() | Sort Key)
{
    $item
}
4

Pour assurer la compatibilité avec les anciennes versions de PowerShell, vous pouvez envisager cette applet de commande:

Function Order-Keys {
    param(
        [Parameter(Mandatory = $true, ValueFromPipeline = $true)][HashTable]$HashTable,
        [Parameter(Mandatory = $false, Position = 1)][ScriptBlock]$Function,
        [Switch]$Descending
    )
    $Keys = $HashTable.Keys | ForEach {$_} # Copy HashTable + KeyCollection
    For ($i = 0; $i -lt $Keys.Count - 1; $i++) {
        For ($j = $i + 1; $j -lt $Keys.Count; $j++) {
            $a = $Keys[$i]
            $b = $Keys[$j]
            If ($Function -is "ScriptBlock") {
                $a = $HashTable[$a] | ForEach $Function
                $b = $HashTable[$b] | ForEach $Function
            }
            If ($Descending) {
                $Swap = $a -lt $b
            }
            Else
            {
                $Swap = $a -gt $b
            }
            If ($Swap) {
                $Keys[$i], $Keys[$j] = $Keys[$j], $Keys[$i]
            }
        }
    }
    Return $Keys
}

Cette applet de commande retourne une liste de clés classées par la définition de fonction:

Trier par nom:

$HashTable | Order-Keys | ForEach {Write-Host $_ $HashTable[$_]}
Germany Berlin
Italy Rome
Spain Madrid
Switzerland Bern

Trier par valeur:

$HashTable | Order-Keys {$_} | ForEach {Write-Host $_ $HashTable[$_]}
Germany Berlin
Switzerland Bern
Spain Madrid
Italy Rome

Vous pouvez également envisager d'imbriquer des tables de hachage:

$HashTable = @{
    Switzerland = @{Order = 1; Capital = "Berne"}
    Germany     = @{Order = 2; Capital = "Berlin"}
    Spain       = @{Order = 3; Capital = "Madrid"}
    Italy       = @{Order = 4; Capital = "Rome"}
}

Par exemple. triez par propriété de commande (hachée) et retournez la clé (pays):

$HashTable | Order-Keys {$_.Order} | ForEach {$_}

Ou triez (par ordre décroissant) selon la capitale prédéfinie:

$HashTable | Order-Keys {$_.Capital} -Descending | ForEach {$_}
1
iRon

La méthode PowerShell 1 consiste à ajouter un membre de table de hachage pour conserver la commande d’ajout. Il n'est pas nécessaire d'utiliser System.Collections.Specialized.OrderedDictionary:

$Hash = New-Object PSObject                                       
$Hash | Add-Member -MemberType NoteProperty -Name key1 -Value val1
$Hash | Add-Member -MemberType NoteProperty -Name key2 -Value val2
$Hash | Add-Member -MemberType NoteProperty -Name key3 -Value val3
1
Axel Limousin
function global:sortDictionaryByKey([hashtable]$dictionary)
{
    return $dictionary.GetEnumerator() | sort -Property name;
}
0
user2174447

Voici une routine simple qui fonctionne pour moi.

function sortedKeys([hashtable]$ht) {
  $out = @()
  foreach($k in $ht.keys) {
    $out += $k
  }
  [Array]::sort($out)
  return ,$out
}

et l'appel à l'utiliser

forEach($k in (& sortedKeys $ht)) {
  ...
}
0
Steve Pritchard