Ruby

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

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

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

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

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

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

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

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

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

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

Рекомендуемые программы