Программирование на языке Ruby
вернуться

Фултон Хэл

Шрифт:

В качестве аналогии можно рассмотреть метод

getline
, который выступает в роли внешнего итератора для объекта класса
IO
. Вы сами вызываете его в удобные моменты времени, а он возвращает прочитанные данные. Сравните это с поведением итератора
each_line
, который последовательно передает программе прочитанные строки.

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

Библиотека

generator
позволяет преобразовать внутренний итератор во внешний. Она предоставляет такие же методы
next
,
rewind
и
end?
, как в классе
IO
. Вот пример:

require 'generator'

array = [7,8,9,10,11,12]

gen = Generator.new(array)

what = gen.current # 7

where = gen.index # 0 (то же, что pos)

while gen.end? and gen.current <11

 gen.next

end

puts gen.current # 11

puts gen.next # 11

puts gen.index # 4 (index - то же, что pos)

puts gen.next? # true (next?
– то же, что end?)

puts gen.next # 12

puts gen.next? # false

Обратите внимание, как мы «читаем» набор по одному элементу в одном или нескольких циклах. Метод

end?
обнаруживает конец набора; если вы проигнорируете его «совет», генератор возбудит исключение
EOFError
. Синонимом
end?
служит
next?
.

Метод

index
(синоним
pos
) сообщает индекс или позицию в наборе. Естественно, индексация начинается с нуля, как в случае с массивом или смещением от начала файла.

Методы

current
и
next
, возможно, интуитивно неочевидны. Представьте себе, что в начале выполняется операция «получить»; тогда текущий (
current
) элемент оказывается таким же, как следующий (
next
). Ясно, что метод next продвигает указатель на следующую позицию, a
current
— нет.

Поскольку для многих наборов возможно только продвижение в прямом направлении, то и генератор ведет себя так же. Не существует метода

prev
(предыдущий); теоретически его можно было бы добавить, но не всегда он был бы применим. Метод
rewind
устанавливает указатель в начало набора.

Недостаток библиотеки

generator
заключается в том, что она реализована с помощью продолжений (
continuation
). Во всех имеющихся на сегодняшний день версиях Ruby это требует большого объема вычислений, поэтому, если итераций много, работа заметно замедляется.

8.4. Заключение

Мы подробно рассмотрели массивы, хэши и перечисляемые структуры в общем. Мы установили определенное сходство между массивами и хэшами, объясняемое тем, что в оба класса подмешан модуль

Enumerable
. Но есть и различия. Мы показали, как преобразовать массив в хэш и наоборот, а также узнали несколько интересных способов расширить стандартное поведение.

Мы изучили различные методы обхода структур, например

each_slice
и
each_cons
, а также выяснили, как работают энумераторы и генераторы.

В главе 9 мы продолжим изучение высокоуровневых структур данных. Не все они входят в ядро Ruby или в стандартные библиотеки. Речь пойдет о множествах, стеках, очередях, деревьях и графах.

Глава 9. Более сложные структуры данных

Графическое представление данных абстрагирует банки памяти любого компьютера. Невообразимая сложность. Лучи света, протянувшиеся в не-пространстве разума, скопления и созвездия данных. Как гаснущие огни большого города.

Вильям Гибсон

Есть, конечно, более сложные и интересные структуры данных, чем массивы и хэши. Некоторые из тех, с которыми мы познакомимся в этой главе, имеют прямую или косвенную поддержку в Ruby, другие приходится программировать самостоятельно. К счастью, Ruby упрощает создание нестандартных структур данных.

Математические множества можно, как мы видели, моделировать с помощью массивов. Но в последних версиях Ruby есть также класс

Set
, который хорошо поддерживает эту структуру.

  • Читать дальше
  • 1
  • ...
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • ...

Private-Bookers - русскоязычная библиотека для чтения онлайн. Здесь удобно открывать книги с телефона и ПК, возвращаться к сохраненной странице и держать любимые произведения под рукой. Материалы добавляются пользователями; если считаете, что ваши права нарушены, воспользуйтесь формой обратной связи.

Моя полка

  • Моя полка

Связаться

  • help@private-bookers.win