Поделиться через


about_Hash_Tables

Краткое описание

Описывает создание, использование и сортировку хэш-хэш-таблицей в PowerShell.

Подробное описание

Хэш-таблица, также известная как словарь или ассоциативный массив, представляет собой компактную структуру данных, которая хранит одну или несколько пар "ключ-значение". Например, хэш-таблица может содержать ряд IP-адресов и имен компьютеров, где IP-адреса являются ключами, а имена компьютеров — значениями или наоборот.

В PowerShell каждая хэш-таблица является объектом типа [System.Collections.Hashtable]. Свойства и методы Hashtable объектов можно использовать в PowerShell.

Начиная с PowerShell 3.0, можно использовать [ordered] ускоритель типов для создания [System.Collections.Specialized.OrderedDictionary] объекта в PowerShell.

Упорядоченные словари отличаются от хэш-списков, что ключи всегда отображаются в порядке их перечисления. Порядок ключей в хэш-файле не детерминирован.

Ключи и значения в хэш-файлах также являются объектами .NET. Они чаще всего являются строками или целыми числами, но они могут иметь любой тип объекта. Можно также создать вложенные хеш-таблицы, где значение ключа — это хеш-таблица.

Хэш-файлы часто используются, так как они эффективны для поиска и извлечения данных. Хэш-списки можно использовать для хранения списков и создания вычисляемых свойств в PowerShell. И командлет ConvertFrom-StringData преобразует структурированные строковые данные в хэш-файл.

Синтаксис

Синтаксис хэш-файла выглядит следующим образом:

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

Синтаксис упорядоченного словаря выглядит следующим образом:

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

Акселератор [ordered] типов появился в PowerShell 3.0.

Чтобы создать хеш-таблицу, выполните следующее пошаговое руководство.

  • Начните хеш-таблицу со знака (@).
  • Заключите хеш-таблицу в фигурные скобки ({}).
  • Введите одну или несколько пар "ключ-значение" для содержимого хэш-файла.
  • Используйте знак равенства (=) для разделения каждого ключа от его значения.
  • Используйте точку с запятой (;) или разрыв строки для разделения пар "ключ-значение".
  • Ключи, содержащие пробелы, должны быть заключены в кавычки. Значения должны быть допустимыми выражениями PowerShell. Строки должны отображаться в кавычках, даже если они не включают пробелы.
  • Чтобы управлять хэш-файлом, сохраните его в переменной.
  • При назначении упорядоченной хеш-таблицы переменной разместите тип [ordered] перед тем, как использовать символ @. Если вы поместите его перед именем переменной, команда завершается ошибкой.

Вы можете использовать упорядоченные словари таким же образом, как и хэштебли. Любой тип можно использовать в качестве значения параметров, которые принимают хэш-таблицу или объекты типа словаря (iDictionary) .

Создание хэш-стилей и упорядоченных словарей

Рассмотрим следующие примеры хеш-таблиц и упорядоченных словарей.

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

Как видно, пары "ключ-значение" в хеш-таблице не представлены в том порядке, в котором они были определены.

Самый простой способ создать упорядоченный словарь — использовать [ordered] атрибут. Поместите атрибут непосредственно перед символом @ .

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

В отличие от хэш-элементов, упорядоченные словари поддерживают порядок значения ключа.

Преобразование хэш-записей и упорядоченных словарей

Нельзя использовать акселератор типов [ordered] для преобразования или приведения хэш-таблицы. Если вы помещете упорядоченный атрибут перед именем переменной, команда завершается ошибкой со следующим сообщением об ошибке.

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

Чтобы исправить выражение, переместите атрибут [упорядочено].

$orderedhash = [ordered]@{}

Вы можете преобразовать упорядоченный словарь в хеш-таблицу, но вы не можете гарантировать порядок элементов.

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

Hashtable и свойства словаря

Хэш-страницы и упорядоченные словари используют несколько свойств. Рассмотрим $hash и $dictionary переменные, определенные в предыдущих примерах.

$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;}

Наиболее используемыми свойствами являются Count, Key, Values и Item.

  • Свойство Count , указывающее количество пар "ключ-значение" в объекте.

  • Свойство ключей — это коллекция имен ключей в хэш-файле или словаре.

    PS> $hash.Keys
    three
    2
    1
    
    PS> $dictionary.Keys
    1
    2
    three
    
  • Свойство Значения представляет собой коллекцию значений в хэш-таблицы или словаре.

    PS> $hash.Values
    3
    two
    one
    
    PS> $dictionary.Values
    one
    two
    3
    
  • Свойство Item — это параметризованное свойство, которое возвращает значение указанного элемента. Хэштебли используют ключ в качестве параметра для параметризованного свойства, а словари используют индекс по умолчанию. Это различие влияет на доступ к значениям для каждого типа.

Доступ к значениям

Существует два распространенных способа доступа к значениям в хэш-таблицы или словаре: нотация элементов или нотация индекса массива.

  • Нотация элемента. К значениям можно получить доступ с помощью имени ключа в качестве свойства члена объекта. Например:

    PS> $hash.1
    one
    
    PS> $dictionary.2
    two
    
  • Нотация индекса массива. К значениям можно получить доступ с помощью нотации индекса. PowerShell преобразует нотацию в вызов параметризованного свойства Item объекта.

    При использовании нотации индекса с хэш-файлами значение внутри квадратных скобок — это имя ключа. Если ключ является строковым значением, заключите имя ключа в кавычки. Например:

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

    В этом примере ключевое значение 2 не является индексом в коллекции значений. Это значение ключа в паре "ключ-значение". Это можно доказать, индексируя в коллекцию значений.

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

    При использовании нотации индекса с словарями значение внутри квадратных скобок интерпретируется на основе его типа. Если значение является целым числом, оно рассматривается как индекс в коллекции значений. Если значение не является целым числом, оно рассматривается как имя ключа. Например:

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

    В этом примере значение [1] массива представляет собой индекс в коллекцию значений с помощью параметризованной Item(int index) перегрузки свойств. Значение [[object]1] массива не является индексом, а значением ключа с помощью перегрузки Item(System.Object key) .

    Примечание.

    Это поведение может быть запутано, если значение ключа является целым числом. По возможности следует избегать использования целых значений ключей в словарях.

Обработка конфликтов имен свойств

Если имя ключа сталкивается с одним из имен свойств типа hashTable HashT able, можно использовать psbaseвстроенных элементов для доступа к этим свойствам. Например, если имя ключа keys и вы хотите вернуть коллекцию ключей HashTable, используйте следующий синтаксис:

$hashtable.psbase.Keys

Это требование применяется для других типов, реализующих System.Collections.IDictionary интерфейс, например OrderedDictionary.

Итерирование ключей и значений

Вы можете произвести итерацию по ключам в хэш-таблице, чтобы обрабатывать значения несколькими способами. Каждый из примеров в этом разделе имеет одинаковые выходные данные. Они выполняют итерацию по переменной, определенной $hash здесь:

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

Примечание.

В этих примерах определяется как упорядоченный словарь, чтобы гарантировать, $hash что выходные данные всегда совпадают. Эти примеры работают одинаково для стандартных хэш-элементов, но порядок выходных данных не предсказуем.

Каждый пример возвращает сообщение для каждого ключа и его значения:

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

В этом примере используется foreach блок для итерации ключей.

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

В этом примере используется ForEach-Object для итерации ключей.

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

В этом примере метод используется для отправки GetEnumerator() каждой пары "ключ-значение" через конвейер ForEach-Object.

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

В этом примере используются GetEnumerator() методы итерации ForEach() по каждой паре "ключ-значение".

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

Добавление и удаление ключей и значений

Как правило, при создании хэш-таблицы в определении включаются пары "ключ-значение". Однако вы можете добавлять и удалять пары "ключ-значение" из хэш-файла в любое время. В следующем примере создается пустая хеш-таблица.

$hash = @{}

Пары "ключ-значение" можно добавлять с помощью нотации массива. Например, в следующем примере добавляется ключ Time со значением Now в хэш-файл.

$hash["Time"] = "Now"

Вы также можете добавить ключи и значения в хэш-таблицы с помощью метода Add() объекта System.Collections.Hashtable. Метод Add() имеет следующий синтаксис:

Add(Key, Value)

Например, чтобы добавить ключ Time со значением Now в хэш-таблицы, используйте следующий формат инструкции.

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

Кроме того, можно добавить ключи и значения в хэш-таблицу с помощью оператора добавления (+) для добавления хэш-таблицы в существующую хэш-таблицу. Например, следующая инструкция добавляет ключ Time со значением Now в хэш-файл в переменной $hash.

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

Можно также добавить значения, хранящиеся в переменных.

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

$hash.Add($t, $now)

Оператор вычитания нельзя использовать для удаления пары "ключ-значение" из хэш-таблицы, но можно использовать метод Remove() хэш-объекта. Метод Remove имеет следующий синтаксис:

$object.Remove(<key>)

В следующем примере удаляется Time пара "ключ-значение" из $hash.

$hash.Remove("Time")

Типы объектов в hashTables

Ключи и значения в хэш-таблицы могут иметь любой тип объекта .NET, а один хэш-файл может иметь ключи и значения нескольких типов.

Следующая инструкция создает хэш-таблицы имен процесса и значения объектов процесса и сохраняет его в переменной $p.

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

Вы можете отобразить хэш-таблицу в $p и использовать свойства имен ключей для отображения значений.

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

Ключи в хэш-таблице могут иметь любой тип .NET. Следующая инструкция добавляет пару "ключ-значение" в хэш-таблицу, связанную с переменной $p. Ключ — это объект службы , представляющий службу WinRM, а значение — текущее состояние службы.

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

Вы можете отобразить и получить доступ к новой паре "ключ-значение", используя те же методы, которые используются для других пар в хэш-файле.

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

Ключи и значения в хэш-таблице также могут быть Hashtable объектами. Следующая инструкция добавляет пару "ключ-значение" в хэш-файл в переменной $p, в которой ключ является строкой, hash2 и значением является хэш-файл с тремя парами "ключ-значение".

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

Вы можете отображать и получать доступ к новым значениям, используя те же методы.

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

Сортировка ключей и значений

Элементы в хэш-таблице неупорядочены. Пары "ключ-значение" могут отображаться в другом порядке при каждом отображении.

Хотя вы не можете сортировать хэш-таблицы, можно использовать метод GetEnumerator() хэш-таблицы для перечисления ключей и значений, а затем использовать командлет Sort-Object для сортировки перечисленных значений для отображения.

Например, следующие команды перечисляют ключи и значения в хэш-таблице в переменной $p , а затем сортируют ключи в алфавитном порядке.

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

Следующая команда использует ту же процедуру для сортировки хэш-значений в порядке убывания.

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

Создание объектов из хэш-элементов

Начиная с PowerShell 3.0, можно создать объект из хэш-таблицы свойств и значений свойств.

Синтаксис выглядит следующим образом:

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

Этот метод работает только для классов, имеющих конструктор без параметров. Свойства объекта должны быть общедоступными и наборными.

Дополнительные сведения см. в about_Object_Creation.

ConvertFrom-StringData

Командлет ConvertFrom-StringData преобразует строку или приведенную здесь строку пар "ключ-значение" в хэш-файл. Командлет можно безопасно использовать ConvertFrom-StringData в разделе "Данные" скрипта, и его можно использовать с Import-LocalizedData командлетом для отображения сообщений пользователей в языке и региональных параметрах пользовательского интерфейса текущего пользователя.

Here-строки особенно полезны, если значения в хэш-таблице включают кавычки. Дополнительные сведения о строках см. в about_Quoting_Rules.

В следующем примере показано, как создать здесь строку сообщений пользователя в предыдущем примере и как использовать ConvertFrom-StringData для преобразования их из строки в хэш-файл.

Следующая команда создает здесь строку пар "ключ-значение", а затем сохраняет ее в переменной $string .

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

Эта команда использует командлет ConvertFrom-StringData для преобразования многострочной строки в хэш-таблицу.

ConvertFrom-StringData $string

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

Дополнительные сведения о строках см. в about_Quoting_Rules.

См. также