mapM для списков
Nov. 25th, 2009 04:53 pm![[personal profile]](https://www.dreamwidth.org/img/silk/identity/user.png)
Читал про монады, наткнулся на несамочевидную шизу с mapM для списков:
mapM (\x -> [x-1, x+1]) [10,20,30]
[ [9,19,29], [9,19,31], [9,21,29], [9,21,31], [11,19,29], [11,19,31], [11,21,29], [11,21,31] ]
Пришлось лезть в ghci и описание устройства mapM, чтобы не сойти с ума.
mapM (\x -> [x-1, x+1]) [10,20,30]
[ [9,19,29], [9,19,31], [9,21,29], [9,21,31], [11,19,29], [11,19,31], [11,21,29], [11,21,31] ]
Пришлось лезть в ghci и описание устройства mapM, чтобы не сойти с ума.
no subject
Date: 2009-11-25 05:23 pm (UTC)но первую минуту идёт дым из головы.
no subject
Date: 2009-11-25 05:37 pm (UTC)Это не mapM для списков, а это поведение монады списков.
<зануда-моде>
Вот ещё вброс:
Prelude> [1, 2] >> [3, 4]
[3,4,3,4]
Prelude> [1, 2, 3] >> [4, 5]
[4,5,4,5,4,5]
Prelude>
;)
Как оно работает: для каждого элемента списка слева (до операции >> или >>=) генерируется список справа. Если это операция (>>), то всё понятно: значения элементов левого списка просто никуда не стучат, ибо не используются аргументом.
В случае (>>=) всё чуть поинтереснее:
Prelude> [1, 2, 3] >>= \x -> [x]
[1,2,3]
Prelude> [1, 2, 3] >>= \x -> [x, x+1]
[1,2,2,3,3,4]
Prelude> [1, 2, 3] >>= \x -> [x, x+100]
[1,101,2,102,3,103]
Prelude>
На последнем примере отчётливо видно, куда деваются элементы списка, идущего первым аргументом в (>>=), и куда девается x+100.
Ну и должно быть очевидно теперь, почему map отличается от mapM === sequence . map:
(Ну или примерно так, чтобы запутать).
no subject
Date: 2009-11-25 05:41 pm (UTC)no subject
Date: 2009-12-13 10:26 pm (UTC)Prelude Control.Monad> replicateM 3 [0,1]
[[0,0,0],[0,0,1],[0,1,0],[0,1,1],[1,0,0],[1,0,1],[1,1,0],[1,1,1]]