about_Hash_Tables
Descrizione breve
Descrive come creare, usare e ordinare tabelle hash in PowerShell.
Descrizione lunga
Una tabella hash, nota anche come dizionario o matrice associativa, è una struttura di dati compatta che archivia una o più coppie chiave-valore. Ad esempio, una tabella hash può contenere una serie di indirizzi IP e nomi di computer, in cui gli indirizzi IP sono le chiavi e i nomi dei computer sono i valori o viceversa.
In PowerShell ogni tabella hash è un oggetto [System.Collections.Hashtable]
. È possibile usare le proprietà e i metodi degli Hashtable oggetti in PowerShell.
A partire da PowerShell 3.0, è possibile usare l'acceleratore [ordered]
di tipi per creare un [System.Collections.Specialized.OrderedDictionary]
oggetto in PowerShell.
I dizionari ordinati differiscono dalle tabelle hash in quanto le chiavi vengono sempre visualizzate nell'ordine in cui vengono elencate. L'ordine delle chiavi in una tabella hash non è deterministico.
Le chiavi e il valore nelle tabelle hash sono anche oggetti .NET. Sono più spesso stringhe o numeri interi, ma possono avere qualsiasi tipo di oggetto. È anche possibile creare tabelle hash annidate, in cui il valore di una chiave è un'altra tabella hash.
Le tabelle hash vengono spesso usate perché sono efficienti per la ricerca e il recupero dei dati. È possibile usare tabelle hash per archiviare elenchi e per creare proprietà calcolate in PowerShell. Il cmdlet ConvertFrom-StringData
converte i dati delle stringhe strutturate in una tabella hash.
Sintassi
La sintassi di una tabella hash è la seguente:
@{ <name> = <value>; [<name> = <value> ] ...}
La sintassi di un dizionario ordinato è la seguente:
[ordered]@{ <name> = <value>; [<name> = <value> ] ...}
L'acceleratore [ordered]
di tipi è stato introdotto in PowerShell 3.0.
Per creare una tabella hash, seguire queste linee guida:
- Iniziare la tabella hash con un segno (
@
). - Racchiudere la tabella hash tra parentesi graffe (
{}
). - Immettere una o più coppie chiave-valore per il contenuto della tabella hash.
- Usare un segno di uguale (
=
) per separare ogni chiave dal relativo valore. - Usare un punto e virgola (
;
) o un'interruzione di riga per separare le coppie chiave-valore. - Le chiavi contenenti spazi devono essere racchiuse tra virgolette. I valori devono essere espressioni di PowerShell valide. Le stringhe devono essere visualizzate tra virgolette, anche se non includono spazi.
- Per gestire la tabella hash, salvatela in una variabile.
- Quando si assegna una tabella hash ordinata a una variabile, posizionare il tipo di
[ordered]
prima del simbolo@
. Se la si posiziona prima del nome della variabile, il comando ha esito negativo.
È possibile usare dizionari ordinati nello stesso modo in cui si usano tabelle hash. Entrambi i tipi possono essere usati come valore di parametri che accettano oggetti di tipo hashtable o dizionario (iDictionary).
Creazione di tabelle hash e dizionari ordinati
Si considerino i seguenti esempi di tabella hash e dizionario ordinato:
$hash = @{
1 = 'one'
2 = 'two'
'three' = 3
}
$hash
Name Value
---- -----
three 3
2 two
1 one
Come si può notare, le coppie chiave-valore in una tabella hash non vengono presentate nell'ordine in cui sono state definite.
Il modo più semplice per creare un dizionario ordinato consiste nell'usare l'attributo [ordered]
. Posizionare l'attributo immediatamente prima del @
simbolo.
$dictionary = [ordered]@{
1 = 'one'
2 = 'two'
'three' = 3
}
$dictionary
Name Value
---- -----
1 one
2 two
three 3
A differenza delle tabelle hash, i dizionari ordinati mantengono l'ordine del valore della chiave.
Conversione di tabelle hash e dizionari ordinati
Non è possibile usare l'acceleratore di tipo [ordered]
per convertire o eseguire il cast di una tabella hash.
Se si inserisce l'attributo ordinato prima del nome della variabile, il comando non riesce con il messaggio di errore seguente.
[ordered]$orderedhash = @{}
ParserError:
Line |
1 | [ordered]$orderedhash = @{}
| ~~~~~~~~~~~~~~
| The ordered attribute can be specified only on a hash literal node.
Per correggere l'espressione, spostare l'attributo [ordered].
$orderedhash = [ordered]@{}
È possibile eseguire il cast di un dizionario ordinato in una tabella hash, ma non è possibile garantire l'ordine dei membri.
[hashtable]$newhash = [ordered]@{
Number = 1
Shape = "Square"
Color = "Blue"
}
$newhash
Name Value
---- -----
Color Blue
Shape Square
Number 1
Hashtable Proprietà del dizionario e
Le tabelle hash e i dizionari ordinati condividono diverse proprietà. Si considerino le $hash
variabili e $dictionary
definite negli esempi precedenti.
$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;}
Le proprietà più usate sono Count, Keys, Values e Item.
Proprietà Count che indica il numero di coppie chiave-valore nell'oggetto .
La proprietà Keys è una raccolta dei nomi delle chiavi nella tabella hash o nel dizionario.
PS> $hash.Keys three 2 1 PS> $dictionary.Keys 1 2 three
La proprietà Values è una raccolta di valori nella tabella hash o nel dizionario.
PS> $hash.Values 3 two one PS> $dictionary.Values one two 3
La proprietà Item è una proprietà con parametri che restituisce il valore dell'elemento specificato. Le tabelle hash usano la chiave come parametro per la proprietà con parametri, mentre i dizionari usano l'indice per impostazione predefinita. Questa differenza influisce sul modo in cui si accede ai valori per ogni tipo.
Accesso ai valori
Esistono due modi comuni per accedere ai valori in una tabella hash o in un dizionario: notazione dei membri o notazione dell'indice di matrice.
Notazione membro: è possibile accedere ai valori usando il nome della chiave come proprietà membro dell'oggetto. Ad esempio:
PS> $hash.1 one PS> $dictionary.2 two
Notazione dell'indice di matrice: è possibile accedere ai valori usando la notazione dell'indice. PowerShell converte tale notazione in una chiamata alla proprietà con parametri Item dell'oggetto .
Quando si usa la notazione dell'indice con tabelle hash, il valore all'interno delle parentesi quadre è il nome della chiave. Se la chiave è un valore stringa, racchiudere il nome della chiave tra virgolette. Ad esempio:
PS> $hash['three'] 3 PS> $hash[2] two
In questo esempio, il valore
2
della chiave non è un indice nella raccolta di valori. È il valore della chiave nella coppia chiave-valore. È possibile dimostrare questa situazione tramite l'indicizzazione nella raccolta di valori.PS> ([array]$hash.Values)[2] one
Quando si usa la notazione di indice con dizionari, il valore all'interno delle parentesi quadre viene interpretato in base al tipo. Se il valore è un numero intero, viene considerato come un indice nella raccolta di valori. Se il valore non è un numero intero, viene considerato come il nome della chiave. Ad esempio:
PS> $dictionary[1] two PS> ([array]$dictionary.Values)[1] two PS> $dictionary[[object]1] one PS> $dictionary['three'] 3
In questo esempio, il valore
[1]
della matrice è un indice nell'insieme di valori usando l'overload dellaItem(int index)
proprietà con parametri. Il valore[[object]1]
della matrice non è un indice, ma un valore chiave usando l'overloadItem(System.Object key)
.Nota
Questo comportamento può generare confusione quando il valore della chiave è un numero intero. Quando possibile, è consigliabile evitare di usare i valori di chiave integer nei dizionari.
Gestione dei conflitti di nomi di proprietà
Se il nome della chiave è in collisione con uno dei nomi di proprietà del tipo hashTable
$hashtable.psbase.Keys
Questo requisito si applica ad altri tipi che implementano l'interfaccia System.Collections.IDictionary , ad esempio OrderedDictionary.
Iterazione su chiavi e valori
È possibile scorrere le chiavi in una tabella hash per elaborare i valori in diversi modi. Ognuno degli esempi di questa sezione ha un output identico. Esegue l'iterazione sulla $hash
variabile definita qui:
$hash = [ordered]@{Number = 1; Shape = "Square"; Color = "Blue"}
Nota
In questi esempi viene $hash
definito come dizionario ordinato per garantire che l'output sia sempre nello stesso ordine. Questi esempi funzionano allo stesso modo per le tabelle hash standard, ma l'ordine dell'output non è prevedibile.
Ogni esempio restituisce un messaggio per ogni chiave e il relativo valore:
The value of 'Number' is: 1
The value of 'Shape' is: Square
The value of 'Color' is: Blue
In questo esempio viene usato un foreach
blocco per scorrere le chiavi.
foreach ($Key in $hash.Keys) {
"The value of '$Key' is: $($hash[$Key])"
}
Questo esempio usa ForEach-Object
per scorrere le chiavi.
$hash.Keys | ForEach-Object {
"The value of '$_' is: $($hash[$_])"
}
In questo esempio viene usato il GetEnumerator()
metodo per inviare ogni coppia chiave-valore tramite la pipeline a ForEach-Object
.
$hash.GetEnumerator() | ForEach-Object {
"The value of '$($_.Key)' is: $($_.Value)"
}
In questo esempio vengono usati i GetEnumerator()
metodi e ForEach()
per scorrere ogni coppia chiave-valore.
$hash.GetEnumerator().ForEach({"The value of '$($_.Key)' is: $($_.Value)"})
Aggiunta e rimozione di chiavi e valori
In genere, quando si crea una tabella hash, si includono le coppie chiave-valore nella definizione. Tuttavia, è possibile aggiungere e rimuovere coppie chiave-valore da una tabella hash in qualsiasi momento. Nell'esempio seguente viene creata una tabella hash vuota.
$hash = @{}
È possibile aggiungere coppie chiave-valore usando la notazione della matrice. Nell'esempio seguente, ad esempio, viene aggiunta una chiave Time
con un valore di Now
alla tabella hash.
$hash["Time"] = "Now"
È anche possibile aggiungere chiavi e valori a una tabella hash usando il metodo Add()
dell'oggetto System.Collections.Hashtable. Il Add()
metodo ha la sintassi seguente:
Add(Key, Value)
Ad esempio, per aggiungere una chiave Time
con un valore di Now
alla tabella hash, usare il formato di istruzione seguente.
$hash.Add("Time", "Now")
È inoltre possibile aggiungere chiavi e valori a una tabella hash usando l'operatore di addizione (+
) per aggiungere una tabella hash a una tabella hash esistente. Ad esempio, l'istruzione seguente aggiunge una chiave Time
con un valore di Now
alla tabella hash nella variabile $hash
.
$hash = $hash + @{Time="Now"}
È anche possibile aggiungere valori archiviati nelle variabili.
$t = "Today"
$now = (Get-Date)
$hash.Add($t, $now)
Non è possibile usare un operatore di sottrazione per rimuovere una coppia chiave-valore da una tabella hash, ma è possibile usare il metodo Remove()
dell'oggetto hashtable. Il Remove
metodo ha la sintassi seguente:
$object.Remove(<key>)
Nell'esempio seguente viene rimossa la Time
coppia chiave-valore da $hash
.
$hash.Remove("Time")
Tipi di oggetto nelle tabelle HashTable
Le chiavi e i valori di una tabella hash possono avere qualsiasi tipo di oggetto .NET e una singola tabella hash può avere chiavi e valori di più tipi.
L'istruzione seguente crea una tabella hash delle stringhe del nome del processo e i valori dell'oggetto processo e la salva nella variabile $p
.
$p = @{
"PowerShell" = (Get-Process PowerShell)
"Notepad" = (Get-Process notepad)
}
È possibile visualizzare la tabella hash in $p
e usare le proprietà key-name per visualizzare i valori.
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
Le chiavi in una tabella hash possono essere di qualsiasi tipo .NET. L'istruzione seguente aggiunge una coppia chiave-valore alla tabella hash nella variabile $p
. La chiave è un oggetto Service che rappresenta il servizio WinRM e il valore è lo stato corrente del servizio.
$p = $p + @{
(Get-Service WinRM) = ((Get-Service WinRM).Status)
}
È possibile visualizzare e accedere alla nuova coppia chiave-valore usando gli stessi metodi usati per altre coppie nella tabella hash.
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
Le chiavi e i valori in una tabella hash possono anche essere Hashtable oggetti . L'istruzione seguente aggiunge una coppia chiave-valore alla tabella hash nella variabile $p
in cui la chiave è una stringa, Hash2 e il valore è una tabella hash con tre coppie chiave-valore.
$p = $p + @{
"Hash2"= @{a=1; b=2; c=3}
}
È possibile visualizzare e accedere ai nuovi valori usando gli stessi metodi.
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
Ordinamento di chiavi e valori
Gli elementi di una tabella hash sono intrinsecamente non ordinati. Le coppie chiave-valore potrebbero essere visualizzate in un ordine diverso ogni volta che vengono visualizzate.
Sebbene non sia possibile ordinare una tabella hash, è possibile usare il metodo GetEnumerator()
delle tabelle hash per enumerare le chiavi e i valori e quindi usare il cmdlet Sort-Object
per ordinare i valori enumerati per la visualizzazione.
Ad esempio, i comandi seguenti enumerare le chiavi e i valori nella tabella hash nella $p
variabile e quindi ordinare le chiavi in ordine alfabetico.
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
Il comando seguente usa la stessa procedura per ordinare i valori hash in ordine decrescente.
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
Creazione di oggetti da tabelle hash
A partire da PowerShell 3.0, è possibile creare un oggetto da una tabella hash di proprietà e valori di proprietà.
La sintassi è la seguente:
[<class-name>]@{
<property-name>=<property-value>
<property-name>=<property-value>
}
Questo metodo funziona solo per le classi con un costruttore senza parametri. Le proprietà dell'oggetto devono essere pubbliche e impostabili.
Per altre informazioni, vedere about_Object_Creation.
ConvertFrom-StringData
Il cmdlet ConvertFrom-StringData
converte una stringa o una stringa here di coppie chiave-valore in una tabella hash. È possibile usare il ConvertFrom-StringData
cmdlet in modo sicuro nella sezione Dati di uno script ed è possibile usarlo con il Import-LocalizedData
cmdlet per visualizzare i messaggi utente nelle impostazioni cultura dell'interfaccia utente dell'utente corrente.
Le here-string sono particolarmente utili quando i valori nell'hash table includono virgolette. Per altre informazioni sulle stringhe qui, vedere about_Quoting_Rules.
Nell'esempio seguente viene illustrato come creare una stringa here dei messaggi utente nell'esempio precedente e come usare ConvertFrom-StringData
per convertirli da una stringa in una tabella hash.
Il comando seguente crea una stringa here delle coppie chiave-valore e quindi la salva nella $string
variabile .
$string = @"
Msg1 = Type "Windows".
Msg2 = She said, "Hello, World."
Msg3 = Enter an alias (or "nickname").
"@
Questo comando usa il cmdlet ConvertFrom-StringData
per convertire la stringa here in una tabella hash.
ConvertFrom-StringData $string
Name Value
---- -----
Msg3 Enter an alias (or "nickname").
Msg2 She said, "Hello, World."
Msg1 Type "Windows".
Per altre informazioni sulle stringhe qui, vedere about_Quoting_Rules.