Изучай Haskell во имя добра! (Липовача) - страница 194

Модуль >Control.Applicative определяет функцию, которая называется >liftA2 и имеет следующий тип:

>liftA2 :: (Applicative f) => (a –> b –> c) –> f a –> f b –> f c

Она определена вот так:

>liftA2 :: (Applicative f) => (a –> b –> c) –> f a –> f b –> f c

>liftA2 f a b = f <$> a <*> b

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

При использовании обычных функторов мы можем просто отображать одно значение функтора с помощью функций. При использовании аппликативных функторов мы можем применять функцию между несколькими значениями функторов. Интересно также рассматривать тип этой функции в виде >(a –> b –> c) –> (f a –> f b –> f c). Когда мы его воспринимаем подобным образом, мы можем сказать, что функция >liftA2 берёт обычную бинарную функцию и преобразует её в функцию, которая работает с двумя аппликативными значениями.

Есть интересная концепция: мы можем взять два аппликативных значения и свести их в одно, которое содержит в себе результаты этих двух аппликативных значений в списке. Например, у нас есть значения >Just 3 и >Just 4. Предположим, что второй функтор содержит одноэлементный список, так как этого очень легко достичь:

>ghci> fmap (\x –> [x]) (Just 4)

>Just [4]

Хорошо, скажем, у нас есть значения >Just 3 и >Just [4]. Как нам получить >Just [3,4]? Это просто!

>ghci> liftA2 (:) (Just 3) (Just [4])

>Just [3,4]

>ghci> (:) <$> Just 3 <*> Just [4]

>Just [3,4]

Вспомните, что оператор >: – это функция, которая принимает элемент и список и возвращает новый список с этим элементом в начале. Теперь, когда у нас есть значение >Just [3,4], могли бы ли мы объединить это со значением >Just 2, чтобы произвести результат >Just [2,3,4]? Да, могли бы. Похоже, мы можем сводить любое количество аппликативных значений в одно, которое содержит список результатов этих аппликативных значений.

Давайте попробуем реализовать функцию, которая принимает список аппликативных значений и возвращает аппликативное значение, которое содержит список в качестве своего результирующего значения. Назовём её >sequenceA:

>sequenceA :: (Applicative f) => [f a] –> f [a]

>sequenceA [] = pure []

>sequenceA (x:xs) = (:) <$> x <*> sequenceA xs

А-а-а, рекурсия! Прежде всего смотрим на тип. Он трансформирует список аппликативных значений в аппликативное значение со списком. После этого мы можем заложить некоторую основу для базового случая. Если мы хотим превратить пустой список в аппликативное значение со списком результатов, то просто помещаем пустой список в контекст по умолчанию. Теперь в дело вступает рекурсия. Если у нас есть список с «головой» и «хвостом» (вспомните,