Введение в Ruby для новичков. Неявное изменение класса при программировании в функциональном стиле

Статья написана студентом Хекслета. Мнение автора может не совпадать с позицией редакции
Читать в полной версии →

Визуально программирование в функциональном стиле выглядит как последовательный вызов методов первоначального объекта. Но в реальности это совсем не так.

Обычно программирование в Ruby в функциональном стиле выглядит так:

str = 'Один Два Три Четыре Пять'
result = str.split(" ").select{|el| !el.start_with?("f")}.map(&:reverse).join(" ")

Визуально такая запись воспринимается как последовательный вызов методов к переменной str (класс String). Фактически, каждый последующий метод вызывается к результату предыдущего. Часто класс результата отличается от первоначального. Следовательно, изменяется и доступный набор методов.

Разберем приведенный выше пример:

str.split(" ") — вызываем метод split для класса String, результат — класс Array

.select{|el| !el.start_with?("f")} — метод select класса Array, результат — новый Array

.map(&:reverse) — метод map класса Array, результат — новый Array

.join(" ") - метод join класса Array, результат - новая строка (String)

В начале и конце — строки (String), но на промежуточных шагах мы работали с массивами (Array) и вызывали методы доступные массивам. Описание методов map и select в документации нужно смотреть для массивов, у строк таких методов нет. Интересно еще, что вызов методов map и select без блока возвращает не массив, а объект класса Enumerator. Он обладает урезанным набором методов по сравнению с массивами.

P.S. Эта заметка написана новичком и для новичков. Понимаю, что для опытных программистов это является очевидным фактом.