The Functor abstraction is used for types that can be mapped over.
___
map f x
/(|>>) x f
/(<<|) f x
/(<!>) f x
static member Map (x:'Functor<'T>, f:'T->'U) :'Functor<'U>
static member Unzip (x:Functor<'T * 'U>) :'Functor<'T> * 'Functor<'U>
map id = id
map (f << g) = map f << map g
Applicative: An applicative is a functor whose map
operation can be splitted in return
and (<*>)
operations,
-
Monad : Monads are functors with an additional Join
operation,
From F#
seq<'T>
list<'T>
array<'T>
'T [,]
'T [,,]
'T [,,,]
option<'T>
IObservable<'T>
Lazy<'T>
Async<'T>
Result<'T,'U>
Choice<'T,'U>
KeyValuePair<'Key,'T>
Map<'Key,'T>
'Monoid * 'T
Task<'T>
'R->'T
Expr<'T>
Dictionary<'Key,'T>
IDictionary<'Key,'T>
IReadOnlyDictionary<'Key,'T>
ResizeArray<'T>
From F#+
Cont<'R,'T>
ContT<'R,'T>
Reader<'R,'T>
ReaderT<'R,'Monad<'T>>
Writer<'Monoid,'T>
WriterT<'Monad<'T * 'Monoid>>
State<'S,'T * 'S>
StateT<'S,'Monad<'T * 'S>>
OptionT<'Monad<option<'T>>
SeqT<'Monad<seq<'T>>
ListT<'Monad<list<'T>>
ResultT<'Monad<Result<'T,'TError>>
ChoiceT<'Monad<Choice<'T,'TError>>
Free<'Functor<'T>,'T>
NonEmptyList<'T>
NonEmptySet<'T>
NonEmptyMap<'Key, 'T>
Validation<'Error,'T>
ZipList<'T>
ParallelArray<'T>
Const<'C,'T>
Compose<'AlternativeF<'AlternativeG<'T>>>
DList<'T>
Kleisli<'T, 'Monad<'U>>
Coproduct<'FunctorL<'T>,'FunctorR<'T>>
Vector<'T,'Dimension>
Matrix<'T,'Rows,'Columns>
Restricted:
- string
- StringBuilder
- Set<'T>
- IEnumerator<'T>
Suggest another concrete implementation
open FSharpPlus
open FSharpPlus.Math.Generic
let getLine = async { return System.Console.ReadLine() }
let putStrLn x = async { printfn "%s" x}
let print x = async { printfn "%A" x}
// Test IO
let action = monad {
do! putStrLn "What is your first name?"
let! fn = getLine
do! putStrLn ("Thanks, " + fn)
do! putStrLn ("What is your last name?")
let! ln = getLine
let fullname = fn + " " + ln
do! putStrLn ("Your full name is: " + fullname)
return fullname }
// Test Functors
let times2,minus3 = (*) 2, (-)/> 3
let resSome1 = map minus3 (Some 4G)
let noValue = map minus3 None
let lstTimes2 = map times2 [1;2;3;4]
let fTimes2minus3 = map minus3 times2
let res39 = fTimes2minus3 21G
let getChars = map (fun (x:string) -> x.ToCharArray() |> Seq.toList ) action
let quot7 = map ((+)2) <@ 5 @>
// try -> runIO getChars ;;
// Define a type Tree
type Tree<'a> =
| Tree of 'a * Tree<'a> * Tree<'a>
| Leaf of 'a
static member map f (t:Tree<'a> ) =
match t with
| Leaf x -> Leaf (f x)
| Tree(x,t1,t2) -> Tree(f x, Tree.map f t1, Tree.map f t2)
// add instance for Functor class
static member Map (x:Tree<_>, f) = Tree.map f x
let myTree = Tree(6, Tree(2, Leaf 1, Leaf 3), Leaf 9)
let mappedTree = map fTimes2minus3 myTree
// An Applicative is automatically a Functor
type ZipList<'s> = ZipList of 's seq with
static member Return (x:'a) = ZipList (Seq.initInfinite (konst x))
static member (<*>) (ZipList (f:seq<'a->'b>), ZipList x) = ZipList (Seq.zip f x |> Seq.map (fun (f, x) -> f x)) : ZipList<'b>
let mappedZipList = map string (ZipList [1;2;3])
// A Monad is automatically a Functor
type MyList<'s> = MyList of 's seq with
static member Return (x:'a) = MyList x
static member (>>=) (MyList x: MyList<'T>, f) = MyList (Seq.collect (f >> (fun (MyList x) -> x)) x)
let mappedMyList = map string (MyList [1;2;3])
Multiple items
module Map
from Microsoft.FSharp.Collections
<summary>Contains operations for working with values of type <see cref="T:Microsoft.FSharp.Collections.FSharpMap`2" />.</summary>
--------------------
type Map<'Key,'Value (requires comparison)> =
interface IReadOnlyDictionary<'Key,'Value>
interface IReadOnlyCollection<KeyValuePair<'Key,'Value>>
interface IEnumerable
interface IComparable
interface IEnumerable<KeyValuePair<'Key,'Value>>
interface ICollection<KeyValuePair<'Key,'Value>>
interface IDictionary<'Key,'Value>
new: elements: seq<'Key * 'Value> -> Map<'Key,'Value>
member Add: key: 'Key * value: 'Value -> Map<'Key,'Value>
member Change: key: 'Key * f: ('Value option -> 'Value option) -> Map<'Key,'Value>
...
<summary>Immutable maps based on binary trees, where keys are ordered by F# generic comparison. By default
comparison is the F# structural comparison function or uses implementations of the IComparable interface on key values.</summary>
<remarks>See the <see cref="T:Microsoft.FSharp.Collections.MapModule" /> module for further operations on maps.
All members of this class are thread-safe and may be used concurrently from multiple threads.</remarks>
--------------------
new: elements: seq<'Key * 'Value> -> Map<'Key,'Value>
val id: x: 'T -> 'T
<summary>The identity function</summary>
<param name="x">The input value.</param>
<returns>The same value.</returns>
<example id="id-example"><code lang="fsharp">
id 12 // Evaulates to 12
id "abc" // Evaulates to "abc"
</code></example>
namespace FSharpPlus
namespace FSharpPlus.Math
module Generic
from FSharpPlus.Math
<summary>
Generic numbers, functions and operators.
By opening this module some common operators become restricted, like (+) to 'T->'T->'T
</summary>
val getLine: Async<string>
val async: AsyncBuilder
<summary>Builds an asynchronous workflow using computation expression syntax.</summary>
<example id="async-1"><code lang="fsharp">
let sleepExample() =
async {
printfn "sleeping"
do! Async.Sleep 10
printfn "waking up"
return 6
}
sleepExample() |> Async.RunSynchronously
</code></example>
namespace System
type Console =
static member Beep: unit -> unit + 1 overload
static member Clear: unit -> unit
static member GetCursorPosition: unit -> struct (int * int)
static member MoveBufferArea: sourceLeft: int * sourceTop: int * sourceWidth: int * sourceHeight: int * targetLeft: int * targetTop: int -> unit + 1 overload
static member OpenStandardError: unit -> Stream + 1 overload
static member OpenStandardInput: unit -> Stream + 1 overload
static member OpenStandardOutput: unit -> Stream + 1 overload
static member Read: unit -> int
static member ReadKey: unit -> ConsoleKeyInfo + 1 overload
static member ReadLine: unit -> string
...
<summary>Represents the standard input, output, and error streams for console applications. This class cannot be inherited.</summary>
System.Console.ReadLine() : string
val putStrLn: x: string -> Async<unit>
val x: string
val printfn: format: Printf.TextWriterFormat<'T> -> 'T
<summary>Print to <c>stdout</c> using the given format, and add a newline.</summary>
<param name="format">The formatter.</param>
<returns>The formatted result.</returns>
<example>See <c>Printf.printfn</c> (link: <see cref="M:Microsoft.FSharp.Core.PrintfModule.PrintFormatLine``1" />) for examples.</example>
val print: x: 'a -> Async<unit>
val x: 'a
val action: Async<string>
val monad<'monad<'t>> : MonadFxBuilder<'monad<'t>>
<summary>
Creates a (lazy) monadic computation expression with side-effects (see http://fsprojects.github.io/FSharpPlus/computation-expressions.html for more information)
</summary>
val fn: string
val ln: string
val fullname: string
val times2: (int -> int)
val minus3: (int -> int)
val resSome1: int option
val map: f: ('T -> 'U) -> x: 'Functor<'T> -> 'Functor<'U> (requires member Map)
<summary>Lifts a function into a Functor.</summary>
<category index="1">Functor</category>
union case Option.Some: Value: 'T -> Option<'T>
<summary>The representation of "Value of type 'T"</summary>
<param name="Value">The input value.</param>
<returns>An option representing the value.</returns>
val noValue: int option
union case Option.None: Option<'T>
<summary>The representation of "No value"</summary>
val lstTimes2: int list
val fTimes2minus3: (int -> int)
val res39: int
val getChars: Async<char list>
Multiple items
val string: value: 'T -> string
<summary>Converts the argument to a string using <c>ToString</c>.</summary>
<remarks>For standard integer and floating point values the and any type that implements <c>IFormattable</c><c>ToString</c> conversion uses <c>CultureInfo.InvariantCulture</c>. </remarks>
<param name="value">The input value.</param>
<returns>The converted string.</returns>
<example id="string-example"><code lang="fsharp"></code></example>
--------------------
type string = System.String
<summary>An abbreviation for the CLI type <see cref="T:System.String" />.</summary>
<category>Basic Types</category>
System.String.ToCharArray() : char[]
System.String.ToCharArray(startIndex: int, length: int) : char[]
Multiple items
module Seq
from FSharpPlus.Operators
--------------------
module Seq
from FSharpPlus
<summary>
Additional operations on Seq
</summary>
--------------------
module Seq
from Microsoft.FSharp.Collections
<summary>Contains operations for working with values of type <see cref="T:Microsoft.FSharp.Collections.seq`1" />.</summary>
val toList: source: seq<'T> -> 'T list
<summary>Builds a list from the given collection.</summary>
<param name="source">The input sequence.</param>
<returns>The result list.</returns>
<exception cref="T:System.ArgumentNullException">Thrown when the input sequence is null.</exception>
<example id="tolist-1"><code lang="fsharp">
let inputs = seq { 1; 2; 5 }
inputs |> Seq.toList
</code>
Evaluates to <c>[ 1; 2; 5 ]</c>.
</example>
val quot7: Quotations.Expr<int>
Multiple items
union case Tree.Tree: 'a * Tree<'a> * Tree<'a> -> Tree<'a>
--------------------
type Tree<'a> =
| Tree of 'a * Tree<'a> * Tree<'a>
| Leaf of 'a
static member Map: x: Tree<'a0> * f: ('a0 -> 'b) -> Tree<'b>
static member map: f: ('a -> 'a0) -> t: Tree<'a> -> Tree<'a0>
union case Tree.Leaf: 'a -> Tree<'a>
val f: ('a -> 'a0)
val t: Tree<'a>
val t1: Tree<'a>
val t2: Tree<'a>
static member Tree.map: f: ('a -> 'a0) -> t: Tree<'a> -> Tree<'a0>
Multiple items
module Map
from FSharpPlus
<summary>
Additional operations on Map<'Key, 'Value>
</summary>
--------------------
module Map
from Microsoft.FSharp.Collections
<summary>Contains operations for working with values of type <see cref="T:Microsoft.FSharp.Collections.FSharpMap`2" />.</summary>
--------------------
type Map<'Key,'Value (requires comparison)> =
interface IReadOnlyDictionary<'Key,'Value>
interface IReadOnlyCollection<KeyValuePair<'Key,'Value>>
interface IEnumerable
interface IComparable
interface IEnumerable<KeyValuePair<'Key,'Value>>
interface ICollection<KeyValuePair<'Key,'Value>>
interface IDictionary<'Key,'Value>
new: elements: seq<'Key * 'Value> -> Map<'Key,'Value>
member Add: key: 'Key * value: 'Value -> Map<'Key,'Value>
member Change: key: 'Key * f: ('Value option -> 'Value option) -> Map<'Key,'Value>
...
<summary>Immutable maps based on binary trees, where keys are ordered by F# generic comparison. By default
comparison is the F# structural comparison function or uses implementations of the IComparable interface on key values.</summary>
<remarks>See the <see cref="T:Microsoft.FSharp.Collections.MapModule" /> module for further operations on maps.
All members of this class are thread-safe and may be used concurrently from multiple threads.</remarks>
--------------------
new: elements: seq<'Key * 'Value> -> Map<'Key,'Value>
val x: Tree<'a>
val f: ('a -> 'b)
val myTree: Tree<int>
val mappedTree: Tree<int>
Multiple items
union case ZipList.ZipList: seq<'s> -> ZipList<'s>
--------------------
type ZipList<'s> =
| ZipList of seq<'s>
static member (<*>) : ZipList<('a -> 'b)> * ZipList<'a> -> ZipList<'b>
static member Return: x: 'a -> ZipList<'a>
Multiple items
val seq: sequence: seq<'T> -> seq<'T>
<summary>Builds a sequence using sequence expression syntax</summary>
<param name="sequence">The input sequence.</param>
<returns>The result sequence.</returns>
<example id="seq-cast-example"><code lang="fsharp">
seq { for i in 0..10 do yield (i, i*i) }
</code></example>
--------------------
type seq<'T> = System.Collections.Generic.IEnumerable<'T>
<summary>An abbreviation for the CLI type <see cref="T:System.Collections.Generic.IEnumerable`1" /></summary>
<remarks>
See the <see cref="T:Microsoft.FSharp.Collections.SeqModule" /> module for further operations related to sequences.
See also <a href="https://docs.microsoft.com/dotnet/fsharp/language-reference/sequences">F# Language Guide - Sequences</a>.
</remarks>
val initInfinite: initializer: (int -> 'T) -> seq<'T>
<summary>Generates a new sequence which, when iterated, will return successive
elements by calling the given function. The results of calling the function
will not be saved, that is the function will be reapplied as necessary to
regenerate the elements. The function is passed the index of the item being
generated.</summary>
<remarks>The returned sequence may be passed between threads safely. However,
individual IEnumerator values generated from the returned sequence should not be accessed concurrently.
Iteration can continue up to <c>Int32.MaxValue</c>.</remarks>
<param name="initializer">A function that generates an item in the sequence from a given index.</param>
<returns>The result sequence.</returns>
<example id="init-infinite-1"><code lang="fsharp">
(+) 5 |> Seq.initInfinite
</code>
Evaluates to a sequence yielding the same results as <c>seq { 5; 6; 7; 8; ... }</c></example>
val konst: k: 'T -> 'Ignored -> 'T
<summary> Creates a constant function.</summary>
<param name="k">The constant value.</param>
<returns>The constant value function.</returns>
<category index="0">Common Combinators</category>
val f: seq<('a -> 'b)>
val x: seq<'a>
val zip: source1: seq<'T1> -> source2: seq<'T2> -> seq<'T1 * 'T2>
<summary>Combines the two sequences into a sequence of pairs. The two sequences need not have equal lengths:
when one sequence is exhausted any remaining elements in the other
sequence are ignored.</summary>
<param name="source1">The first input sequence.</param>
<param name="source2">The second input sequence.</param>
<returns>The result sequence.</returns>
<exception cref="T:System.ArgumentNullException">Thrown when either of the input sequences is null.</exception>
<example id="zip-1"><code lang="fsharp">
let numbers = [1; 2]
let names = ["one"; "two"]
Seq.zip numbers names
</code>
Evaluates to a sequence yielding the same results as <c>seq { (1, "one"); (2, "two") }</c>.
</example>
val map: mapping: ('T -> 'U) -> source: seq<'T> -> seq<'U>
<summary>Builds a new collection whose elements are the results of applying the given function
to each of the elements of the collection. The given function will be applied
as elements are demanded using the <c>MoveNext</c> method on enumerators retrieved from the
object.</summary>
<remarks>The returned sequence may be passed between threads safely. However,
individual IEnumerator values generated from the returned sequence should not be accessed concurrently.</remarks>
<param name="mapping">A function to transform items from the input sequence.</param>
<param name="source">The input sequence.</param>
<returns>The result sequence.</returns>
<exception cref="T:System.ArgumentNullException">Thrown when the input sequence is null.</exception>
<example id="item-1"><code lang="fsharp">
let inputs = ["a"; "bbb"; "cc"]
inputs |> Seq.map (fun x -> x.Length)
</code>
Evaluates to a sequence yielding the same results as <c>seq { 1; 3; 2 }</c></example>
val mappedZipList: ZipList<string>
Multiple items
union case MyList.MyList: seq<'s> -> MyList<'s>
--------------------
type MyList<'s> =
| MyList of seq<'s>
static member (>>=) : MyList<'T> * f: ('T -> MyList<'a>) -> MyList<'a>
static member Return: x: #seq<'a0> -> MyList<'a0>
val x: #seq<'a0>
val x: seq<'T>
val f: ('T -> MyList<'a>)
val collect: mapping: ('T -> #seq<'U>) -> source: seq<'T> -> seq<'U>
<summary>Applies the given function to each element of the sequence and concatenates all the
results.</summary>
<remarks>Remember sequence is lazy, effects are delayed until it is enumerated.</remarks>
<param name="mapping">A function to transform elements of the input sequence into the sequences
that will then be concatenated.</param>
<param name="source">The input sequence.</param>
<returns>The result sequence.</returns>
<exception cref="T:System.ArgumentNullException">Thrown when the input sequence is null.</exception>
<example id="collect-1"><code lang="fsharp">
type Foo = { Bar: int seq }
let input = seq { {Bar = [1; 2]}; {Bar = [3; 4]} }
input |> Seq.collect (fun foo -> foo.Bar)
</code>
Evaluates to a sequence yielding the same results as <c>seq { 1; 2; 3; 4 }</c></example>
<example id="collect-2"><code lang="fsharp">
let input = [[1; 2]; [3; 4]]
input |> Seq.collect id
</code>
Evaluates to a sequence yielding the same results as <c>seq { 1; 2; 3; 4 }</c></example>
val mappedMyList: MyList<char>