This library offers a set of computation expressions for conveniently and efficiently constructing common collections.
Get it on NuGet: FSharp.Collections.Builders.
dotnet add package FSharp.Collections.Builders
Or try it in an F# script file (.fsx
):
#r "nuget: FSharp.Collections.Builders"
open FSharp.Collections.Builders
See the API documentation for the full set of supported collections and operations.
The computation expression builders exposed by this library support most of the operations available in the built-in
list, array, and sequence expressions, including for
, while
, yield!
, try
/with
, try
/finally
, and conditionals.1
Open FSharp.Collections.Builders
for type-inference-friendly builders that behave similarly to the built-in list,
array, and sequence expressions in that they treat any collection used with for
or yield!
as a _ seq
.
open FSharp.Collections.Builders
The builders in this namespace don't require any type annotations in situations like this:
// No need to add a type annotation to xs;
// xs is inferred to have type 'a seq.
let f xs =
resizeArray {
for x in xs -> string x
}
Enables efficiently constructing and transforming instances of ResizeArray<'T>
(System.Collections.Generic.List<'T>
).
let xs = resizeArray { 1; 2; 3 }
Compiles to:
let xs = ResizeArray () xs.Add 1 xs.Add 2 xs.Add 3
let ys = resizeArray { for x in xs -> float x * 2.0 }
Compiles to:
let ys = ResizeArray () for x in xs do ys.Add (float x * 2.0)
let f xs = resizeArray { for x in xs do if x % 3 = 0 then x }
xs
isint seq
.
let g xs = resizeArray<float> { for x in xs -> x * x }
xs
isfloat seq
.
Enables efficiently constructing and transforming instances of System.Collections.Generic.HashSet<'T>
.
let xs = hashSet { 1; 2; 3 }
hashSet [1; 2; 3]
let ys = hashSet { yield! xs; yield! xs }
hashSet [1; 2; 3]
Enables efficiently constructing and transforming instances of System.Collections.Generic.SortedSet<'T>
.
let xs = sortedSet { 3; 2; 1; 2 }
sortedSet [1; 2; 3]
Enables efficiently constructing and transforming instances of System.Collections.Generic.Dictionary<'TKey, 'TValue>
.
let kvs = dictionary { 1, "a"; 2, "b"; 3, "c" }
let filtered = dictionary { for k, v in kvs do if k > 1 then k, v }
let list = [1..100]
let stringMap = dictionary { for i in list -> string i, i }
Enables efficiently constructing and transforming instances of System.Collections.Generic.SortedDictionary<'TKey, 'TValue>
.
let m = sortedDictionary { 2, "b"; 3, "c"; 1, "a"; 3, "d" }
sortedDictionary [1, "a"; 2, "b"; 3, "d"]
The set'
2 and map
builders enable ergonomically constructing immutable F# sets and maps
without requiring intermediate collections or the ceremony of folds or recursive functions.
let xs = set' { 1; 2; 3 }
set [1; 2; 3]
let ys = set' { 1; 2; 3; 3 }
set [1; 2; 3]
let xs = [5; 1; 1; 1; 3; 3; 2; 2; 5; 5; 5]
let ys = set' { for x in xs do if x &&& 1 <> 0 then x }
set [1; 3; 5]
let m = map { 2, "b"; 3, "c"; 1, "a"; 3, "d" }
map [1, "a"; 2, "b"; 3, "d"]
let m = map { for x in 1..100 -> x, x * x }
Equivalent to
let m = Map.ofSeq (seq { for x in 1..100 -> x, x * x })or
let m = (Map.empty, {1..100}) ||> Seq.fold (fun m x -> m.Add (x, x * x))
Enables efficiently constructing and transforming instances of System.Collections.Immutable.ImmutableArray<'T>
.
let xs = immutableArray { 1; 2; 3 }
Compiles to:
let xs = let builder = ImmutableArray.CreateBuilder () builder.Add 1 builder.Add 2 builder.Add 3 builder.ToImmutable ()
let ys = immutableArray { for x in xs -> string x }
let xs = immutableList { 1; 2; 3 }
immutableList [1; 2; 3]
let xs = immutableHashSet { 3; 1; 2; 3 }
immutableHashSet [3; 1; 2]
let xs = immutableSortedSet { 3; 1; 2; 3 }
immutableSortedSet [1; 2; 3]
let kvs = immutableDictionary { 1, "a"; 2, "b"; 3, "c" }
immutableDictionary [1, "a"; 2, "b"; 3, "c"]
let kvs = immutableSortedDictionary { 2, "b"; 3, "c"; 1, "a"; 3, "d" }
immutableSortedDictionary [1, "a"; 2, "b"; 3, "d"]
Given
let xs = [1..100]
Instead of
let s =
xs
|> List.filter (fun x -> x % 3 = 0)
|> List.sum
sum
enables:
let s = sum { for x in xs do if x % 3 = 0 then x }
The Greek capital letter sigma Σ
may read better in certain domains:
Σ { for item in items -> item.Subtotal } <= 0.10 * total
To enable specialized overloads of for
and yield!
that give increased iteration performance
at the cost of requiring that the type of the collection being iterated be statically known:
open FSharp.Collections.Builders.Specialized
let f (xs : int list) =
resizeArray {
for x in xs -> x * x
}
Compiles down to a fast
'T list
iteration.
let g xs =
let len = Array.length xs
resizeArray {
for x in xs -> x * len
}
Compiles down to a fast integer
for
-loop.
F#'s built-in list,
array,
and sequence expressions
make initializing and transforming 'T list
, 'T array
, and 'T seq
quite nice:
let nums = [1; 2; 3]
let doubled = [ for num in nums -> float num * 2.0 ]
let doubledArr = [| for num in nums -> float num * 2.0 |]
let doubledSeq = seq { for num in nums -> float num * 2.0 }
But when it comes time to work with one of the common mutable collection types from System.Collections.Generic
,
whether to interoperate with other .NET libraries or for specific performance or modeling reasons, F# doesn't
provide the same syntactic sugar, forcing you you to switch modes from expression-based to statement-based style:
let nums = ResizeArray ()
nums.Add 1
nums.Add 2
nums.Add 3
Or, to keep the ergonomics of sequence expressions, you must instantiate and iterate over intermediate collections:
let nums = ResizeArray [1; 2; 3]
let nums = ResizeArray (seq { 1; 2; 3 })
C# offers collection initialization syntax for types that implement IEnumerable<T>
and have a public Add
method:
var nums = new List<int> { 1, 2, 3 };
var nums = new HashSet<int> { 1, 2, 3 };
Or, with target-typed new()
:
List<int> nums = new() { 1, 2, 3 };
HashSet<int> nums = new() { 1, 2, 3 };
F# 6's resumable code feature makes it straightforward to implement efficient, ergonomic computation expression builders for collection types that aren't special-cased by the F# compiler.
This library implements such builders for several common mutable and immutable collections from System.Collections.Generic
and System.Collections.Immutable
, and FSharp.Collections
,
as well as offering generic collection
and dict'
builders that support any collection type with a default constructor and an appropriate Add
method.
There are a couple language suggestions that might someday (happily!) make parts of this library obsolete:
Additional potential development directions:
- Add versions that take initial capacities, equality comparers, etc.
- Add a vectorized version of the
sum
expression.
Footnotes
-
It is not yet possible to provide custom implementations of the range operator
(..)
for computation expression builders—although there is an approved language suggestion for it. ↩ -
Alas, unlike
seq
, which is special-cased by the compiler and can act as both a function and computation expression builder, it is impossible to do the same with the built-inset
function (without it being similarly special-cased in the compiler). ↩