type'aMaybe=Justof'a|Nothing(* Functor *)type'aMaybewithstaticmemberfmap():('a->'b)->'aMaybe->'bMaybe=funf->function|Justx->fx|>Just|Nothing->Nothing(* Applicative *)type'aMaybewithstaticmemberliftA():('a->'b)Maybe->'aMaybe->'bMaybe=funfm->funm->matchfm,mwith|Justf,Justx->fx|>Just|______________->Nothing(* Monad *)type'aMaybewithstaticmemberliftM():'aMaybe->('a->'bMaybe)->'bMaybe=funm->funf->matchmwith|Nothing->Nothing|Justx->fx(* Maybe with functions, the amount of parenthesis is to damn high *)fmap((+)1)(Just42);;liftA(Just((+)1))(Just42);;liftM(Just42)(funx->x+1|>Just);;(* Maybe with operators, fewer parenthesis *)((+)1)<@>Just42;;Just((+)1)<*>Just42;;Just42>>=funx->x+1|>Just;;
Maybe (Option) with fmap, liftA and liftM Code output:
>
type 'a Maybe =
| Just of 'a
| Nothing
with
static member fmap : unit -> (('a -> 'b) -> 'a Maybe -> 'b Maybe)
static member liftA : unit -> (('a -> 'b) Maybe -> 'a Maybe -> 'b Maybe)
static member liftM : unit -> ('a Maybe -> ('a -> 'b Maybe) -> 'b Maybe)
end
> val it : int Maybe = Just 43
> val it : int Maybe = Just 43
> val it : int Maybe = Just 43
> val it : int Maybe = Just 43
> val it : int Maybe = Just 43
> val it : int Maybe = Just 43
Vect (list) with fmap, liftA and liftM Code Snippet
(* A list in F# is just a type abbreviation of a FSharpList *)typedefof<List<_>>=typedefof<_list>(* Functor *)type'aListwithstaticmemberfmap():('a->'b)->'alist->'blist=List.map(* This works *)List.fmap()((+)1)[0..10](* but this doesn't *)//fmap((+)1)<@>[42](* error FS0001: The type ''a list' does not support the operator 'fmap' *)(* Therefore, lets create our own type wrapping native lists in a Vector *)type'aVect=Vectof'alist(* Functor *)type'aVectwithstaticmemberfmap():('a->'b)->'aVect->'bVect=funf->fun(Vectxs)->List.mapfxs|>Vect(* Applicative *)type'aVectwithstaticmemberliftA():('a->'b)Vect->'aVect->'bVect=fun(Vectfs)->fun(Vectxs)->fs|>List.map(funf->xs|>List.mapf)|>List.concat|>Vect(* Monad *)type'aVectwithstaticmemberliftM():'aVect->('alist->'bVect)->'bVect=fun(Vectxs)->funf->fxs(* Vect with operators, fewer parenthesis *)((+)1)<@>Vect[0..5];;Vect[id;((+)1)]<*>Vect[0..5];;Vect[0..5]>>=funxs->xs|>List.map((+)1)|>Vect;;
Vect (list) with fmap, liftA and liftM Code output:
>
type List<'T> with
static member fmap : unit -> (('T -> 'b) -> 'T list -> 'b list)
type 'a Vect =
| Vect of 'a list
with
static member fmap : unit -> (('a -> 'b) -> 'a Vect -> 'b Vect)
static member liftA : unit -> (('a -> 'b) Vect -> 'a Vect -> 'b Vect)
static member liftM : unit -> ('a Vect -> ('a list -> 'b Vect) -> 'b Vect)
end
> val it : int Vect = Vect [1; 2; 3; 4; 5; 6]
> val it : int Vect = Vect [0; 1; 2; 3; 4; 5; 1; 2; 3; 4; 5; 6]
> val it : int Vect = Vect [1; 2; 3; 4; 5; 6]
Result (Choice) with fmap, liftA and liftM Code Snippet
type('error,'a)Result=Okof'a|Errof'error(* Functor *)type('c,'a)Resultwithstaticmemberfmap():('a->'b)->('c,'a)Result->('c,'b)Result=funf->function|Okx->fx|>Ok|Erre->Erre(* Applicative *)type('c,'a)ResultwithstaticmemberliftA():('c,('a->'b))Result->('c,'a)Result->('c,'b)Result=funfm->funm->matchfm,mwith|Okf,Okx->fx|>Ok|Erre,_|_,Erre->Erre(* Monad *)type('c,'a)ResultwithstaticmemberliftM():('c,'a)Result->('a->('c,'b)Result)->('c,'b)Result=funm->funf->matchmwith|Erre->Erre|Okx->fx(* Result with operators, fewer parenthesis *)(((+)1)<@>Ok42:(exn,int)Result);;(Ok((+)1)<*>Ok42:(exn,int)Result);;(Ok42>>=(funx->x+1|>Ok):(exn,int)Result);;
Result (Choice) with fmap, liftA and liftM Code output:
>
type ('error,'a) Result =
| Ok of 'a
| Err of 'error
with
static member
fmap : unit -> (('a -> 'b) -> ('c,'a) Result -> ('c,'b) Result)
static member
liftA : unit ->
(('c,('a -> 'b)) Result -> ('c,'a) Result -> ('c,'b) Result)
static member
liftM : unit ->
(('c,'a) Result -> ('a -> ('c,'b) Result) -> ('c,'b) Result)
end
> val it : (exn,int) Result = Ok 43
> val it : (exn,int) Result = Ok 43
> val it : (exn,int) Result = Ok 43
Combining Maybe and Result using (»=) Code Snippet
Combining Maybe and Result using (»=) Code output:
> val inc : x:int -> int
> val incM : x:int -> int Maybe
> val defM : x:'a -> _arg1:'a Maybe -> 'a
> val incR : x:int -> ('a,int) Result
> val defR : _arg1:(#System.Exception,'b) Result -> 'b
> val it : int = 50