Как бы вы ни представляли себе хэш, это полезный и мощный инструмент программирования.
8.2.1. Создание нового хэша
Как и в случае с классом Array
, для создания хэша служит специальный метод класса []
. Данные, перечисленные в квадратных скобках, образуют ассоциированные пары. Ниже показаны шесть способов вызвать этот метод (все хэши с a1
до c2
содержат одни и те же данные).
a1 = Hash.[]('flat',3,'curved',2)
a2 = Hash.[]('flat'=>3,'curved'=>2)
b1 = Hash['flat',3,'curved',2]
b2 = Hash['flat'=>3,'curved'=>2]
c1 = {'flat',3,'curved',2}
c2 = {'flat'=>3,'curved'=>2}
# Для a1, b1 и c1 число элементов должно быть четным.
Есть также метод new
, который может принимать параметр, задающий значение по умолчанию. Отметим, что это значение не является частью хэша — оно просто используется вместо nil
.
d = Hash.new # Создать пустой хэш.
е = Hash.new(99) # Создать пустой хэш.
f = Hash.new('а'=>3) # Создать пустой хэш.
е['angled'] # 99
e.inspect # {}
f['b'] # {'а'=>3} (значением по умолчанию
# является тоже хэш).
f.inspect # {}
8.2.2. Указание значения по умолчанию для хэша
Значением по умолчанию для хэша является объект, возвращаемый вместо nil
в случае, когда указанный ключ не найден. Это полезно, если вы планируете вызывать для возвращенного значения методы, которые для nil
не определены. Задать значение по умолчанию можно в момент создания хэша или позже с помощью метода default=
.
Все отсутствующие ключи указывают на один и тот же объект по умолчанию, поэтому изменение данного объекта имеет побочный эффект.
а = Hash.new('missing') # Объект по умолчанию - строка 'missing'.
a['hello'] # 'missing'
а.default='nothing'
a['hello'] # 'nothing'
a['good'] << 'bye' # 'nothingbye'
a.default # 'nothingbye'
Имеется также специальный метод экземпляра fetch
, который возбуждает исключение IndexError
, если в объекте типа Hash
нет указанного ключа. Он принимает также второй параметр, играющий роль значения по умолчанию. Кроме того, методу fetch
можно передать необязательный блок, который выработает значение по умолчанию, если ключ не будет найден. Таким образом, каждому отсутствующему ключу можно сопоставить свое «значение по умолчанию».
а = {'flat',3,'curved',2,'angled',5}
a.fetch('pointed') # IndexError
a.fetch('curved','na') # 2
a.fetch('x','na') # 'na'
a.fetch('flat') {|x| x.upcase} # 3
a.fetch('pointed') {|x| x.upcase) # 'POINTED'
8.2.3. Доступ к парам ключ-значение и добавление новых пар
В классе Hash
есть методы класса []
и []=
. Используются они почти так же, как одноименные методы в классе Array
, но принимают лишь один параметр. В качестве параметра может выступать любой объект, а не только строка (хотя строки используются чаще всего).
а = {}
а['flat'] = 3 # {'flat'=>3}
а.[]=('curved',2) # {'flat'=>3,'curved'=>2}
a.store('angled',5) # {'flat'=>3,'curved'=>2,'angled'=>5}
Метод store
— просто синоним []=
, оба могут принимать два аргумента, как показано в примере выше.
Метод fetch
аналогичен методу []
, но возбуждает исключение IndexError
, когда ключ отсутствует. Есть у него и необязательный второй аргумент (или блок) для указания значения по умолчанию (см. раздел 8.2.2).
a['flat'] # 3
а.[]('flat') # 3
a.fetch('flat') # 3
a['bent'] # nil
Предположим, что мы не уверены, существует ли объект Hash
, но хотели бы избежать очистки имеющегося хэша. Очевидное решение — проверить, определен ли интересующий нас объект:
unless defined? а
а={}
end
a['flat'] = 3
Но есть и другой способ:
а ||= {}
a['flat'] = 3
# Или даже так:
(а ||= {})['flat'] = 3
Тот же вопрос можно поставить для отдельных ключей, когда новое значение следует присваивать, лишь если такого ключа еще нет:
a=Hash.new(99)
а[2] # 99
а # {}
а[2] ||= 5 # 99
а # {}
b=Hash.new