Функции высшего порядка — это основной способ обработки коллекций во многих языках. И Ruby здесь не исключение. Идиоматический код на Ruby выглядит, как цепочка методов, преобразующих коллекцию в какой-то нужный выход. Такая обработка строится на базе методов map()
, filter()
, reduce()
и их более специфичных аналогах.
Краткий обзор:
# Обходит коллекцию и вызывает для каждого элемента блок
# Результат складывает в новую коллекцию, которая возвращается
[1, 2, 3].map { |v| v ** 2 } # [1, 4, 9]
# Фильтрует коллекцию от элементов не удовлетворяющих условию
[1, 2, 3].filter { |v| v.odd? } # [1, 3]
# Обходит коллекцию, накапливая в процессе аккумулятор (acc)
# После обработки каждого элемента возврат из блока становится
# новым аккумулятором. В конце он возвращается наружу
# Начальное значение аккумулятора передается первым аргументом
[1, 2, 3].reduce(0) { |acc, v| acc + v } # 6
При таком подходе вся обработка разбивается на множество небольших действий, которые комбинируются друг с другом, превращаясь в цепочку. Фактически это функциональное программирование на объектах. Оно популярно в большинстве современных языков, поэтому здесь не разбирается подробно.
Рассмотрим несколько примеров разных преобразований, которые можно выполнять с помощью этих функций с данными, приближенными к реальным.
users = [
{ name: 'Bronn', gender: 'male', birthday: '1973-03-23' },
{ name: 'Reigar', gender: 'male', birthday: '1973-11-03' },
{ name: 'Tisha', gender: 'female', birthday: '2012-11-03' },
{ name: 'Sansa', gender: 'female', birthday: '2012-11-03' },
{ name: 'Rick', gender: 'male', birthday: '2012-11-03' },
]
# Формируем список пользователей младше 30 лет
users.filter { |u| Time.now.year - Time.new(u[:birthday]).year < 30 }
# Формируем список тех же пользователей, но без gender
# Обратите внимание на двойные фигурные скобки —
# внешние для блока, внутренние для хеша
users.map { |u| { name: u[:name], birthday: u[:birthday] } }
# Группируем пользователей по полу
# Аналогичного результата можно добиться с помощью group_by
users.reduce({}) do |acc, u|
acc[u[:gender]] ||= []
acc[u[:gender]] << u
acc # обязательно вернуть новый acc
end
Реализуйте функцию get_men_count_by_year()
, которая принимает список пользователей и возвращает объект, где ключ – это год рождения, а значение – это количество мужчин, родившихся в этот год.
users = [
{ name: 'Bronn', gender: 'male', birthday: '1973-03-23' },
{ name: 'Reigar', gender: 'male', birthday: '1973-11-03' },
{ name: 'Eiegon', gender: 'male', birthday: '1963-11-03' },
{ name: 'Sansa', gender: 'female', birthday: '2012-11-03' },
{ name: 'Jon', gender: 'male', birthday: '1980-11-03' },
{ name: 'Robb', gender: 'male', birthday: '1980-05-14' },
{ name: 'Tisha', gender: 'female', birthday: '2012-11-03' },
{ name: 'Rick', gender: 'male', birthday: '2012-11-03' },
{ name: 'Joffrey', gender: 'male', birthday: '1999-11-03' },
{ name: 'Edd', gender: 'male', birthday: '1973-11-03' }
]
get_men_count_by_year(users)
# {
# '1973' => 3,
# '1963' => 1,
# '1980' => 2,
# '2012' => 1,
# '1999' => 1
# }
Если вы зашли в тупик, то самое время задать вопрос в «Обсуждениях». Как правильно задать вопрос:
Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.
Это нормально 🙆, в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи.
В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.
Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в «Обсуждениях». Идеально, если вы сформулируете непонятные моменты в виде вопросов. Обычно нам нужно несколько дней для внесения правок.
Кстати, вы тоже можете участвовать в улучшении курсов: внизу есть ссылка на исходный код уроков, который можно править прямо из браузера.
Ваше упражнение проверяется по этим тестам
1# frozen_string_literal: true
2
3require 'test_helper'
4require_relative 'index'
5
6describe 'function' do
7 it 'should works' do
8 users = [
9 { name: 'Bronn', gender: 'male', birthday: '1973-03-23' },
10 { name: 'Reigar', gender: 'male', birthday: '1973-11-03' },
11 { name: 'Eiegon', gender: 'male', birthday: '1963-11-03' },
12 { name: 'Sansa', gender: 'female', birthday: '2012-11-03' },
13 { name: 'Jon', gender: 'male', birthday: '1980-11-03' },
14 { name: 'Robb', gender: 'male', birthday: '1980-05-14' },
15 { name: 'Tisha', gender: 'female', birthday: '2012-11-03' },
16 { name: 'Rick', gender: 'male', birthday: '2012-11-03' },
17 { name: 'Joffrey', gender: 'male', birthday: '1999-11-03' },
18 { name: 'Edd', gender: 'male', birthday: '1973-11-03' }
19 ]
20
21 expected = {
22 '1973' => 3,
23 '1963' => 1,
24 '1980' => 2,
25 '2012' => 1,
26 '1999' => 1
27 }
28
29 assert { get_men_count_by_year(users) == expected }
30 end
31end
32
Решение учителя откроется через: