Бесплатный курс по Ruby. Зарегистрируйтесь для отслеживания прогресса →

Ruby: Метод fetch и извлечение значений из хеша

В Ruby значения из хеша по ключу извлекает метод [] и fetch(). В чем же принципиальное отличие между ними?

Если запрашивать значение по несуществующему ключу, метод [] молча возвращает nil.

env = {
  host: 'localhost',
  port: 3000 
}
puts env[:user] # -> nil

Метод fetch() же действует иначе - если ключа нет, получим ошибку КеуError.

env = {
  host: 'localhost',
  port: 3000 
}
puts env.fetch(:user) # ->  получим ошибку key not found (KeyError)

Но это поведение можно переопределить, так как метод fetch() принимает вторым параметром значение по умолчанию:

env = {
  host: 'localhost',
  port: 3000 
}
puts env.fetch(:user, nil) # -> nil
puts env.fetch(:user, 'guest') # ->  'guest'

И даже блок или лямбда-функцию:

env = { host: 'localhost' }

env.fetch(:user) {  |key| env[key] = 'guest' }
puts env # -> { host: 'localhost', user: 'guest' }

Пример конфигурации почтовой рассылки из приложения на Ruby on Rails:

config.action_mailer.smtp_settings = {
  user_name: ENV.fetch('MAIL_USERNAME', nil),
  password: ENV.fetch('MAIL_PASSWORD', nil),
  address: ENV.fetch('MAIL_HOST', nil),
  domain: ENV.fetch('MAIL_HOST', nil),
  port: ENV.fetch('SMTP_PORT', '25'),
  authentication: :cram_md5
}

Ruby style guide советует использовать метод fetch() вместо метода [].

Задание

Реализуйте функцию setup_env().

Функция принимает: ключ, хеш и блок который тоже принимает два параметра: хеш и ключ из параметров функции.

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

Eсли ключ присутствует, то функция должна вернуть хеш в неизменном виде.

env = { api_key: 123 }
setup_env(:host, env) { |env, key|  env[key] = 'localhost' } # -> { api_key: 123, host: 'localhost' }
setup_env(:api_key, env) { |env, key|  env[key] = 'localhost' } # -> { api_key: 123 }
env == { api_key: 123 } # -> true

Функция не должна мутировать исходный объект.

Упражнение не проходит проверку — что делать? 😶

Если вы зашли в тупик, то самое время задать вопрос в «Обсуждениях». Как правильно задать вопрос:

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

Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.

Мой код отличается от решения учителя 🤔

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

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

Прочитал урок — ничего не понятно 🙄

Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в «Обсуждениях». Идеально, если вы сформулируете непонятные моменты в виде вопросов. Обычно нам нужно несколько дней для внесения правок.

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

Полезное


Нашли ошибку? Есть что добавить? Пулреквесты приветствуются https://github.com/hexlet-basics
Если вы столкнулись с трудностями и не знаете, что делать, задайте вопрос в нашем большом и дружном сообществе