Delen via


about_Hash_Tables

Korte beschrijving

Hierin wordt beschreven hoe u hashtabellen maakt, gebruikt en sorteert in PowerShell.

Lange beschrijving

Een hashtable, ook wel een woordenlijst of associatieve matrix genoemd, is een compacte gegevensstructuur waarin een of meer sleutel-waardeparen worden opgeslagen. Een hashtabel kan bijvoorbeeld een reeks IP-adressen en computernamen bevatten, waarbij de IP-adressen de sleutels zijn en de computernamen de waarden zijn, of omgekeerd.

In PowerShell is elk hashtable een [System.Collections.Hashtable] object. U kunt de eigenschappen en methoden van Hashtable objecten in PowerShell gebruiken.

Vanaf PowerShell 3.0 kunt u de [ordered] typeversneller gebruiken om een [System.Collections.Specialized.OrderedDictionary] object te maken in PowerShell.

Geordende woordenlijsten verschillen van hashtabellen omdat de sleutels altijd worden weergegeven in de volgorde waarin u ze opgeeft. De volgorde van sleutels in een hashtable is niet deterministisch.

De sleutels en waarde in hashtabellen zijn ook .NET-objecten. Dit zijn meestal tekenreeksen of gehele getallen, maar ze kunnen elk objecttype hebben. U kunt ook geneste hashtabellen maken, waarin de waarde van een sleutel een andere hashtableis.

Hashtables worden vaak gebruikt omdat ze efficiënt zijn voor het zoeken en ophalen van gegevens. U kunt hashtabellen gebruiken om lijsten op te slaan en berekende eigenschappen te maken in PowerShell. En met de ConvertFrom-StringData cmdlet worden gestructureerde tekenreeksgegevens geconverteerd naar een hashtable.

Syntaxis

De syntaxis van een hashtable is als volgt:

@{ <name> = <value>; [<name> = <value> ] ...}

De syntaxis van een geordende woordenlijst is als volgt:

[ordered]@{ <name> = <value>; [<name> = <value> ] ...}

De [ordered] typeversneller is geïntroduceerd in PowerShell 3.0.

Volg deze richtlijnen om een hashtablete maken:

  • hashtable Begin met een bijteken (@).
  • Plaats de hashtable accolades ({}).
  • Voer een of meer sleutel-waardeparen in voor de inhoud van de hashtable.
  • Gebruik een gelijkteken (=) om elke sleutel te scheiden van de waarde.
  • Gebruik een puntkomma (;) of een regeleinde om de sleutel-waardeparen te scheiden.
  • Sleutels die spaties bevatten, moeten tussen aanhalingstekens worden geplaatst. Waarden moeten geldige PowerShell-expressies zijn. Tekenreeksen moeten tussen aanhalingstekens worden weergegeven, zelfs als ze geen spaties bevatten.
  • Als u de variabele wilt beheren, slaat u deze hashtableop in een variabele.
  • Wanneer u een geordende variabele hashtable toewijst, plaatst u het [ordered] type vóór het @ symbool. Als u deze vóór de naam van de variabele plaatst, mislukt de opdracht.

U kunt geordende woordenlijsten op dezelfde manier gebruiken als u hashtables gebruikt. Beide typen kunnen worden gebruikt als de waarde van parameters die een hashtable of woordenlijstobjecteniDictionary () gebruiken.

Hashtables en geordende woordenlijsten maken

Bekijk de volgende hashtable en geordende woordenlijstvoorbeelden:

$hash = @{
    1       = 'one'
    2       = 'two'
    'three' = 3
}
$hash
Name                           Value
----                           -----
three                          3
2                              two
1                              one

Zoals u kunt zien, worden de sleutel-waardeparen in een hashtable niet weergegeven in de volgorde waarin ze zijn gedefinieerd.

De eenvoudigste manier om een geordende woordenlijst te maken, is door het [ordered] kenmerk te gebruiken. Plaats het kenmerk direct vóór het @ symbool.

$dictionary = [ordered]@{
    1       = 'one'
    2       = 'two'
    'three' = 3
}
$dictionary
Name                           Value
----                           -----
1                              one
2                              two
three                          3

In tegenstelling tot hashtables behouden geordende woordenlijsten de volgorde van de sleutelwaarde.

Hashtables en geordende woordenlijsten converteren

U kunt de [ordered] typeversneller niet gebruiken om een hashtable. Als u het geordende kenmerk vóór de naam van de variabele plaatst, mislukt de opdracht met het volgende foutbericht.

[ordered]$orderedhash = @{}
ParserError:
Line |
   1 |  [ordered]$orderedhash = @{}
     |  ~~~~~~~~~~~~~~
     | The ordered attribute can be specified only on a hash literal node.

Als u de expressie wilt corrigeren, verplaatst u het kenmerk [geordende] .

$orderedhash = [ordered]@{}

U kunt een geordende woordenlijst casten naar een hashtable, maar u kunt de volgorde van de leden niet garanderen.

[hashtable]$newhash = [ordered]@{
    Number = 1
    Shape = "Square"
    Color = "Blue"
}
$newhash
Name                           Value
----                           -----
Color                          Blue
Shape                          Square
Number                         1

Hashtable en woordenlijsteigenschappen

Hashtables en geordende woordenlijsten delen verschillende eigenschappen. Houd rekening met de $hash variabelen die $dictionary zijn gedefinieerd in de vorige voorbeelden.

$hash | Get-Member -MemberType Properties, ParameterizedProperty
   TypeName: System.Collections.Hashtable

Name           MemberType            Definition
----           ----------            ----------
Item           ParameterizedProperty System.Object Item(System.Object key) {get;set;}
Count          Property              int Count {get;}
IsFixedSize    Property              bool IsFixedSize {get;}
IsReadOnly     Property              bool IsReadOnly {get;}
IsSynchronized Property              bool IsSynchronized {get;}
Keys           Property              System.Collections.ICollection Keys {get;}
SyncRoot       Property              System.Object SyncRoot {get;}
Values         Property              System.Collections.ICollection Values {get;}
$dictionary | Get-Member -MemberType Properties, ParameterizedProperty
   TypeName: System.Collections.Specialized.OrderedDictionary

Name           MemberType            Definition
----           ----------            ----------
Item           ParameterizedProperty System.Object Item(int index) {get;set;},
                                     System.Object Item(System.Object key) {get;set;}
Count          Property              int Count {get;}
IsFixedSize    Property              bool IsFixedSize {get;}
IsReadOnly     Property              bool IsReadOnly {get;}
IsSynchronized Property              bool IsSynchronized {get;}
Keys           Property              System.Collections.ICollection Keys {get;}
SyncRoot       Property              System.Object SyncRoot {get;}
Values         Property              System.Collections.ICollection Values {get;}

De meest gebruikte eigenschappen zijn Count, Keys, Values en Item.

  • De eigenschap Count die het aantal sleutel-waardeparen in het object aangeeft.

  • De eigenschap Sleutels is een verzameling sleutelnamen in de hashtable of woordenlijst.

    PS> $hash.Keys
    three
    2
    1
    
    PS> $dictionary.Keys
    1
    2
    three
    
  • De eigenschap Waarden is een verzameling van de waarden in de hashtable of woordenlijst.

    PS> $hash.Values
    3
    two
    one
    
    PS> $dictionary.Values
    one
    two
    3
    
  • De eigenschap Item is een geparameteriseerde eigenschap die de waarde retourneert van het item dat u opgeeft. Hashtables gebruiken de sleutel als parameter voor de geparameteriseerde eigenschap, terwijl woordenlijsten de index standaard gebruiken. Dit verschil is van invloed op de manier waarop u toegang krijgt tot de waarden voor elk type.

Toegang tot waarden

Er zijn twee veelvoorkomende manieren om toegang te krijgen tot de waarden in een hashtable of woordenlijst: notatie van leden of matrixindex notatie.

  • Lid-notatie : waarden kunnen worden geopend met behulp van de sleutelnaam als lideigenschap van het object. Voorbeeld:

    PS> $hash.1
    one
    
    PS> $dictionary.2
    two
    
  • Matrixindex notatie : waarden kunnen worden geopend met behulp van index notatie. PowerShell converteert die notatie naar een aanroep naar de eigenschap Itemparameter van het object.

    Wanneer u index notatie met hashtables gebruikt, is de waarde binnen de vierkante haken de sleutelnaam. Als de sleutel een tekenreekswaarde is, plaatst u de sleutelnaam tussen aanhalingstekens. Voorbeeld:

    PS> $hash['three']
    3
    
    PS> $hash[2]
    2
    

    In dit voorbeeld is de sleutelwaarde 2 geen index in de verzameling waarden. Dit is de waarde van de sleutel in het sleutel-waardepaar. U kunt dit bewijzen door te indexeren in de verzameling waarden.

    PS> ([array]$hash.Values)[2]
    one
    

    Wanneer u index notatie met woordenlijsten gebruikt, wordt de waarde binnen de vierkante haken geïnterpreteerd op basis van het type. Als de waarde een geheel getal is, wordt deze beschouwd als een index in de verzameling waarden. Als de waarde geen geheel getal is, wordt deze behandeld als de sleutelnaam. Voorbeeld:

    PS> $dictionary[1]
    two
    PS> ([array]$dictionary.Values)[1]
    two
    PS> $dictionary[[object]1]
    one
    PS> $dictionary['three']
    3
    

    In dit voorbeeld is de matrixwaarde [1] een index in de verzameling waarden met behulp van de Item(int index) overbelaste eigenschap met parameters. De matrixwaarde [[object]1] is geen index, maar een sleutelwaarde die gebruikmaakt van de Item(System.Object key) overbelasting.

    Notitie

    Dit gedrag kan verwarrend zijn wanneer de sleutelwaarde een geheel getal is. Indien mogelijk moet u het gebruik van sleutelwaarden voor gehele getallen in woordenlijsten vermijden.

Conflicten tussen eigenschapsnamen afhandelen

Als de sleutelnaam conflicteert met een van de eigenschapsnamen van het HashTable type, kunt u het intrinsieke psbase-lid gebruiken om toegang te krijgen tot deze eigenschappen. Als de sleutelnaam bijvoorbeeld is keys en u de verzameling van de HashTable sleutels wilt retourneren, gebruikt u deze syntaxis:

$hashtable.psbase.Keys

Deze vereiste is van toepassing op andere typen die de System.Collections.IDictionary interface implementeren, zoals OrderedDictionary.

Herhalen van sleutels en waarden

U kunt de sleutels hashtable op verschillende manieren herhalen om de waarden te verwerken. Elk van de voorbeelden in deze sectie heeft identieke uitvoer. Ze herhalen de $hash variabele die hier is gedefinieerd:

$hash = [ordered]@{ Number = 1; Shape = "Square"; Color = "Blue"}

Notitie

In deze voorbeelden $hash wordt gedefinieerd als een geordende woordenlijst om ervoor te zorgen dat de uitvoer altijd in dezelfde volgorde staat. Deze voorbeelden werken hetzelfde voor standaard-hashtabellen, maar de volgorde van de uitvoer is niet voorspelbaar.

Elk voorbeeld retourneert een bericht voor elke sleutel en de bijbehorende waarde:

The value of 'Number' is: 1
The value of 'Shape' is: Square
The value of 'Color' is: Blue

In dit voorbeeld wordt een foreach blok gebruikt om de sleutels te herhalen.

foreach ($Key in $hash.Keys) {
    "The value of '$Key' is: $($hash[$Key])"
}

In dit voorbeeld wordt gebruikgemaakt ForEach-Object van het herhalen van de sleutels.

$hash.Keys | ForEach-Object {
    "The value of '$_' is: $($hash[$_])"
}

In dit voorbeeld wordt de GetEnumerator() methode gebruikt om elk sleutel-waardepaar via de pijplijn naar ForEach-Objectte verzenden.

$hash.GetEnumerator() | ForEach-Object {
    "The value of '$($_.Key)' is: $($_.Value)"
}

In dit voorbeeld worden de GetEnumerator() en ForEach() methoden gebruikt om elk sleutel-waardepaar te herhalen.

$hash.GetEnumerator().ForEach({"The value of '$($_.Key)' is: $($_.Value)"})

Sleutels en waarden toevoegen en verwijderen

Wanneer u een hashtable sleutel-waardeparen maakt in de definitie, neemt u meestal de sleutel-waardeparen op. U kunt echter op elk gewenst moment sleutel-waardeparen toevoegen aan en verwijderen.hashtable In het volgende voorbeeld wordt een lege hashtable.

$hash = @{}

U kunt sleutel-waardeparen toevoegen met behulp van matrix-notatie. In het volgende voorbeeld wordt bijvoorbeeld een Time sleutel met een waarde aan Now de hashtablesleutel toegevoegd.

$hash["Time"] = "Now"

U kunt ook sleutels en waarden toevoegen aan een hashtable met behulp van de Add() methode van het System.Collections.Hashtable object. De Add() methode heeft de volgende syntaxis:

Add(Key, Value)

Als u bijvoorbeeld een Time sleutel met een waarde aan Now de hashtablesleutel wilt toevoegen, gebruikt u de volgende instructieindeling.

$hash.Add("Time", "Now")

En u kunt sleutels en waarden toevoegen aan een hashtable met behulp van de optellingsoperator (+) om een hashtable aan een bestaande hashtabletoe te voegen. Met de volgende instructie wordt bijvoorbeeld een Time sleutel met een waarde aan Now de hashtable $hash variabele toegevoegd.

$hash = $hash + @{Time="Now"}

U kunt ook waarden toevoegen die zijn opgeslagen in variabelen.

$t = "Today"
$now = (Get-Date)

$hash.Add($t, $now)

U kunt geen aftrekkingsoperator gebruiken om een sleutel-waardepaar uit een hash-tabel te verwijderen, maar u kunt de Remove() methode van het hashtable object gebruiken. De Remove methode heeft de volgende syntaxis:

$object.Remove(<key>)

In het volgende voorbeeld wordt het Time sleutel-waardepaar verwijderd uit $hash.

$hash.Remove("Time")

Objecttypen in HashTables

De sleutels en waarden in een hashtable kunnen elk .NET-objecttype hebben en één hashtable kan sleutels en waarden van meerdere typen bevatten.

Met de volgende instructie maakt u een hashtable reeks tekenreeksen voor procesnamen en objectwaarden en slaat u deze op in de $p variabele.

$p = @{
    "PowerShell" = (Get-Process PowerShell)
    "Notepad" = (Get-Process notepad)
}

U kunt de eigenschappen $p van de hashtable sleutelnaam weergeven en gebruiken om de waarden weer te geven.

PS> $p

Name                           Value
----                           -----
PowerShell                     System.Diagnostics.Process (PowerShell)
Notepad                        System.Diagnostics.Process (notepad)

PS> $p.PowerShell

Handles  NPM(K)    PM(K)      WS(K) VM(M)   CPU(s)     Id ProcessName
-------  ------    -----      ----- -----   ------     -- -----------
    441      24    54196      54012   571     5.10   1788 PowerShell

PS> $p.keys | ForEach-Object {$p.$_.handles}
441
251

De sleutels in een hashtable kunnen elk .NET-type zijn. Met de volgende instructie wordt een sleutel-waardepaar aan de hashtable $p variabele toegevoegd. De sleutel is een serviceobject dat de WinRM-service vertegenwoordigt en de waarde de huidige status van de service is.

$p = $p + @{
    (Get-Service WinRM) = ((Get-Service WinRM).Status)
}

U kunt het nieuwe sleutel-waardepaar weergeven en openen met dezelfde methoden die u gebruikt voor andere paren in de hashtable.

PS> $p

Name                           Value
----                           -----
PowerShell                     System.Diagnostics.Process (PowerShell)
Notepad                        System.Diagnostics.Process (notepad)
System.ServiceProcess.Servi... Running

PS> $p.keys
PowerShell
Notepad

Status   Name               DisplayName
------   ----               -----------
Running  winrm              Windows Remote Management (WS-Manag...

PS> $p.keys | ForEach-Object {$_.name}
WinRM

De sleutels en waarden in een hashtable kunnen ook objecten zijn Hashtable . De volgende instructie voegt sleutel-waardepaar toe aan de hashtable $p variabele waarin de sleutel een tekenreeks, Hash2 en de waarde een hashtable met drie sleutel-waardeparen is.

$p = $p + @{
    "Hash2"= @{a=1; b=2; c=3}
}

U kunt de nieuwe waarden weergeven en openen met dezelfde methoden.

PS> $p

Name                           Value
----                           -----
PowerShell                     System.Diagnostics.Process (pwsh)
Hash2                          {[a, 1], [b, 2], [c, 3]}
Notepad                        System.Diagnostics.Process (Notepad)
WinRM                          Running

PS> $p.Hash2

Name                           Value
----                           -----
a                              1
b                              2
c                              3

PS> $p.Hash2.b
2

Sleutels en waarden sorteren

De items in een hashtable zijn intrinsiek ongeordeerd. De sleutel-waardeparen worden mogelijk in een andere volgorde weergegeven telkens wanneer u ze weergeeft.

Hoewel u een hashtableniet kunt sorteren, kunt u de GetEnumerator() methode van hashtables gebruiken om de sleutels en waarden op te sommen en vervolgens de Sort-Object cmdlet gebruiken om de geïnventareerde waarden voor weergave te sorteren.

Met de volgende opdrachten worden bijvoorbeeld de sleutels en waarden in de hash-tabel in de $p variabele opgesomd en worden de sleutels vervolgens in alfabetische volgorde gesorteerd.

PS> $p.GetEnumerator() | Sort-Object -Property key

Name                           Value
----                           -----
Hash2                          {[a, 1], [b, 2], [c, 3]}
Notepad                        System.Diagnostics.Process (Notepad)
PowerShell                     System.Diagnostics.Process (pwsh)
WinRM                          Running

De volgende opdracht gebruikt dezelfde procedure om de hashwaarden in aflopende volgorde te sorteren.

PS> $p.GetEnumerator() | Sort-Object -Property Value -Descending

Name                           Value
----                           -----
PowerShell                     System.Diagnostics.Process (pwsh)
Notepad                        System.Diagnostics.Process (Notepad)
Hash2                          {[a, 1], [b, 2], [c, 3]}
WinRM                          Running

Objecten maken op afstand van hashtabellen

Vanaf PowerShell 3.0 kunt u een object maken op basis van eigenschappen hashtable en eigenschapswaarden.

De syntaxis is als volgt:

[<class-name>]@{
  <property-name>=<property-value>
  <property-name>=<property-value>
}

Deze methode werkt alleen voor klassen die een constructor hebben die geen parameters heeft. De objecteigenschappen moeten openbaar en ingesteld zijn.

Zie about_Object_Creation voor meer informatie.

ConvertFrom-StringData

De ConvertFrom-StringData cmdlet converteert een tekenreeks of een here-tekenreeks van sleutel-waardeparen naar een hashtable. U kunt de ConvertFrom-StringData cmdlet veilig gebruiken in de sectie Gegevens van een script en u kunt deze gebruiken met de Import-LocalizedData cmdlet om gebruikersberichten weer te geven in de gebruikersinterfacecultuur van de huidige gebruiker.

Hier zijn tekenreeksen vooral handig wanneer de waarden in de hashtable aanhalingstekens tussen aanhalingstekens staan. Zie about_Quoting_Rules voor meer informatie over deze tekenreeksen.

In het volgende voorbeeld ziet u hoe u een hier-tekenreeks maakt van de gebruikersberichten in het vorige voorbeeld en hoe u deze kunt converteren ConvertFrom-StringData van een tekenreeks naar een hashtable.

Met de volgende opdracht maakt u een here-string van de sleutel-waardeparen en slaat u deze vervolgens op in de $string variabele.

$string = @"
Msg1 = Type "Windows".
Msg2 = She said, "Hello, World."
Msg3 = Enter an alias (or "nickname").
"@

Met deze opdracht wordt de ConvertFrom-StringData cmdlet gebruikt om de here-string te converteren naar een hashtable.

ConvertFrom-StringData $string

Name                           Value
----                           -----
Msg3                           Enter an alias (or "nickname").
Msg2                           She said, "Hello, World."
Msg1                           Type "Windows".

Zie about_Quoting_Rules voor meer informatie over deze tekenreeksen.

Zie ook