В Real World Haskell, глава 4. " Функциональное программирование":
Напишите фолд с фолдом:
-- file: ch04/Fold.hs
myFoldl :: (a -> b -> a) -> a -> [b] -> a
myFoldl f z xs = foldr step id xs z
where step x g a = g (f a x)
Приведенный выше код меня сильно смутил, и кто-то по имени dps переписал его со значимым именем, чтобы сделать его немного понятнее:
myFoldl stepL zeroL xs = (foldr stepR id xs) zeroL
where stepR lastL accR accInitL = accR (stepL accInitL lastL)
Кто-то еще, Джеф Г, отлично поработал, предоставив пример и шаг за шагом продемонстрировав базовый механизм:
myFoldl (+) 0 [1, 2, 3]
= (foldR step id [1, 2, 3]) 0
= (step 1 (step 2 (step 3 id))) 0
= (step 1 (step 2 (\a3 -> id ((+) a3 3)))) 0
= (step 1 (\a2 -> (\a3 -> id ((+) a3 3)) ((+) a2 2))) 0
= (\a1 -> (\a2 -> (\a3 -> id ((+) a3 3)) ((+) a2 2)) ((+) a1 1)) 0
= (\a1 -> (\a2 -> (\a3 -> (+) a3 3) ((+) a2 2)) ((+) a1 1)) 0
= (\a1 -> (\a2 -> (+) ((+) a2 2) 3) ((+) a1 1)) 0
= (\a1 -> (+) ((+) ((+) a1 1) 2) 3) 0
= (+) ((+) ((+) 0 1) 2) 3
= ((0 + 1) + 2) + 3
Но я до сих пор не могу до конца понять, вот мои вопросы:
- Для чего нужна функция id? Какова роль? Зачем нам это нужно здесь?
- В приведенном выше примере функция id является аккумулятором в лямбда-функции?
- Прототип foldr - это
foldr :: (a -> b -> b) -> b -> [a] -> b
, и первый параметр - это функция, которой нужны два параметра, но функция шага в реализации myFoldl использует 3 параметра, я совершенно запутался!