Аналогичным образом, записав fmap :: (a –> b)
–>
(f
a
–>
f
b)
, мы можем воспринимать fmap
не как функцию, которая принимает одну функцию и значение функтора и возвращает значение функтора, но как функцию, которая принимает функцию и возвращает новую функцию, которая такая же, как и прежняя, за исключением того, что она принимает значение функтора в качестве параметра и возвращает значение функтора в качестве результата. Она принимает функцию типа a –> b
и возвращает функцию типа f a –> f b
. Это называется :t
в GHCi:
ghci> :t fmap (*2)
fmap (*2) :: (Num a, Functor f) => f a –> f a
ghci> :t fmap (replicate 3)
fmap (replicate 3) :: (Functor f) => f a –> f [a]
Выражение fmap (*2)
– это функция, которая получает функтор f
над числами и возвращает функтор над числами. Таким функтором могут быть список, значение Maybe
, Either String
или что-то другое. Выражение fmap (replicate 3)
получит функтор над любым типом и вернёт функтор над списком элементов данного типа. Это становится ещё очевиднее, если мы частично применим, скажем, fmap
(++"!")
, а затем привяжем её к имени в GHCi.
Вы можете рассматривать fmap
двояко:
• как функцию, которая принимает функцию и значение функтора, а затем отображает это значение функтора с помощью данной функции;
• как функцию, которая принимает функцию и втягивает её в функтор, так чтобы она оперировала значениями функторов.
Обе точки зрения верны.
Тип fmap (replicate 3) :: (Functor f) => f a –> f [a]
означает, что функция будет работать с любым функтором. Что именно она будет делать, зависит от функтора. Если мы применим fmap
(replicate 3)
к списку, будет выбрана реализация fmap
для списка, то есть просто map
. Если мы применим её к Maybe a
, она применит replicate 3
к значению внутри Just
. Если это значение равно Nothing
, то оно останется равным Nothing
. Вот несколько примеров:
ghci> fmap (replicate 3) [1,2,3,4]
[[1,1,1],[2,2,2],[3,3,3],[4,4,4]]
ghci> fmap (replicate 3) (Just 4)
Just [4,4,4]
ghci> fmap (replicate 3) (Right "ля")
Right ["ля","ля","ля"]
ghci> fmap (replicate 3) Nothing
Nothing
ghci> fmap (replicate 3) (Left "фуу")
Left "фуу"
Законы функторов
Предполагается, что все функторы проявляют определённые свойства и поведение. Они должны надёжно вести себя как сущности, которые можно отобразить. Применение функции fmap
к функтору должно только отобразить функтор с помощью функции – ничего более. Это поведение описано в законах функторов. Все экземпляры класса Functor
должны следовать этим двум законам. Язык Haskell не принуждает, чтобы эти законы выполнялись автоматически, поэтому вы должны проверять их сами, когда создаёте функтор. Все экземпляры класса Functor
в стандартной библиотеке выполняют эти законы.
Закон 1
Первый закон функторов гласит, что если мы применяем функцию id
к значению функтора, то значение функтора, которое мы получим, должно быть таким же, как первоначальное значение функтора. В формализованной записи это выглядит так: fmap id = id
. Иными словами, если мы применим fmap id
к значению функтора, это должно быть то же самое, что и просто применение функции id
к значению. Вспомните, что id
– это функция тождества, которая просто возвращает свой параметр неизменным. Она также может быть записана в виде \x –> x
. Если воспринимать значение функтора как нечто, что может быть отображено, то закон fmap id = id
представляется довольно очевидным.
Давайте посмотрим, выполняется ли он для некоторых значений функторов:
ghci> fmap id (Just 3)
Just 3
ghci> id (Just 3)
Just 3
ghci> fmap id [1..5]
[1,2,3,4,5]
ghci> id [1..5]
[1,2,3,4,5]
ghci> fmap id []
[]
ghci> fmap id Nothing
Nothing
Если посмотреть на реализацию функцию fmap
, например, для типа Maybe
, мы можем понять, почему выполняется первый закон функторов:
instance Functor Maybe where
fmap f (Just x) = Just (f x)
fmap f Nothing= Nothing