52700.fb2 Учебник по Haskell - читать онлайн бесплатно полную версию книги . Страница 97

Учебник по Haskell - читать онлайн бесплатно полную версию книги . Страница 97

generate 0 f = idK

generate n f = f *> generate (n - 1) f

Или мы можем воспользоваться функцией iterate и написать это определение так:

generate :: Int -> (a -> [a]) -> (a -> [a])

generate n f = iterate (*> f) idK !! n

Функция iterate принимает функцию вычисления следующего элемента и начальное значение и строит

бесконечный список итераций:

iterate :: (a -> a) -> a -> [a]

iterate f a = [a, f a, f (f a), f (f (f a)), ... ]

Если мы подставим наши аргументы то мы получим список:

[id, f, f*> f, f*> f*> f, f*> f*> f*> f, ... ]

Проверим как работает эта функция в интерпретаторе. Для этого мы сначала дополним наш модуль

Kleisli определением экземпляра для списков и функциями next и generate:

*Kleisli> :reload

[2 of 2] Compiling Kleisli

( Kleisli. hs, interpreted )

Ok, modules loaded: Kleisli, Nat.

*Kleisli> let gen n = generate n next ’a’

*Kleisli> gen 0

”a”

92 | Глава 6: Функторы и монады: теория

*Kleisli> gen 1

”ab”

*Kleisli> gen 2

”aba”

*Kleisli> gen 3

”abaab”

*Kleisli> gen 4

”abaababa”

Правила L-системы задаются многозначной функцией. Функция generate позволяет по такой функции

строить произвольное поколение развития буквенного организма.

6.3 Применение функций

Давайте определим в терминах композиции ещё одну полезную функцию. А именно функцию примене-

ния. Вспомним её тип:

($) :: (a -> b) -> a -> b

Эту функцию можно определить через композицию, если у нас есть в наличии постоянная функция и

единичный тип. Мы будем считать, что константа это функция из единичного типа в значение. Превратив

константу в функцию мы можем составить композицию:

($) :: (a -> b) -> a -> b

f $ a = (const a >> f) ()

В самом конце мы подставляем специальное значение (). Это значение единичного типа (unit type) или

кортежа с нулём элементов. Единичный тип имеет всего одно значение, которым мы и воспользовались в

этом определении. Зачем такое запутанное определение, вместо привычного (f a)? Оказывается точно таким

же способом мы можем определить применение в нашем мире специальных функций a -> m b.

Применение в этом мире происходит особенным образом. Необходимо помнить о том, что второй аргу-

мент функции применения, значение, которое мы подставляем в функцию, также было получено из какой-то

другой функции. Поэтому оно будет иметь такую же форму, что и значения справа от стрелки. В нашем

случае это m b.

Посмотрим на типы специальных функций применения:

(*$) :: (a -> m b) -> m a -> m b