Ruby: Трансляция символов в блоки

В Ruby-коде можно встретить странную конструкцию из амперсанда, соединенного с символом:

['hexlet', 'code-basics'].map(&:upcase) # ["HEXLET", "CODE-BASICS"]
# То же самое
# ['hexlet', 'code-basics'].map { |name| name.upcase }

Амперсанд в этом выражении обозначает передачу блока в функцию. Но символ — это не блок. Как работает такой код? Всё дело в приведении типов. У символов определен метод to_proc(), который преобразует символ в блок определенного вида. Он вызывается автоматически в тех случаях, когда данные используются как блоки. Это то же самое, что и интерполяция данных в строку.

В отличие от простых типов данных, преобразование симола в блок работает не очевидно. Проще показать на примере:

block = :capitalize.to_proc
# proc = { |value| value.capitalize }
block.call('hexlet') # "Hexlet"

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

['hexlet', 'code-basics'].map(&:reverse).map(&:capitalize)
# ["Telxeh", "Scisab-edoc"]

Этот трюк работает даже для операторов, так как в Ruby большинство операторов всего лишь методы:

[1, 3, 4].reduce &:+ # 8

Задание

Реализуйте функцию, которая принимает на вход список строк, и применяет к нему следующую цепочку действий:

  • Список сортируется
  • Фильтрация: удаляются строки, не заканчивающиеся на '?'
  • Отображение: все строки приводятся к нижнему регистру

Результат возвращается наружу:

strings = ['wow?', 'One?', 'tWo!', 'THREE']
convert(strings) # ["one?", "wow?"]

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