ArrayT 0.26.0
dotnet add package ArrayT --version 0.26.0
NuGet\Install-Package ArrayT -Version 0.26.0
<PackageReference Include="ArrayT" Version="0.26.0" />
<PackageVersion Include="ArrayT" Version="0.26.0" />
<PackageReference Include="ArrayT" />
paket add ArrayT --version 0.26.0
#r "nuget: ArrayT, 0.26.0"
#:package ArrayT@0.26.0
#addin nuget:?package=ArrayT&version=0.26.0
#tool nuget:?package=ArrayT&version=0.26.0

ArrayT
ArrayT is an F# extension and module library for Array<'T>
It also works in Javascript and Typescript with Fable.
Motivation
I was always annoyed that an IndexOutOfRangeException does not include the actual index that was out of bounds nor the actual size of the array.
This library fixes that in array.Get, array.Set, array.Slice and other item access functions.
This library was designed for use with F# scripting.
Functions and methods never return null.
Only functions starting with try... will return an F# Option.
Otherwise when a function fails on invalid input it will throw a descriptive exception.
See also https://github.com/goswinr/ResizeArray/ for a similar library for ResizeArray<'T>.
It Includes
An
Arraymodule that has a additional functions to theArraymodule fromFSharp.Core.
See docsExtension members on
Arraylike
.Get(idx).Set(idx,item).First.Last.SecondLastxs.DebugIdx.[i]and more..
With nicer IndexOutOfRangeExceptions that include the bad index and the actual size.
See docs
Full API Documentation
Usage
Just open the namespace
open ArrayT
this namespace contains:
a module also called
Array. It will add additional functions to theArraymodule fromFSharp.Core.this will also auto open the extension members on
Array<'T>
Examples
Better error messages
The core motivation: descriptive exceptions when accessing out-of-range indices.
#r "nuget: ArrayT"
open ArrayT
let xs = [| 0 .. 88 |]
xs.Get(99)
throws
System.IndexOutOfRangeException: Array.Get: Can't get index 99 from:
array<Int32> with 89 items:
0: 0
1: 1
2: 2
3: 3
4: 4
...
88: 88
instead of the usual unhelpful System.IndexOutOfRangeException: Index was outside the bounds of the array.
The same applies to Set:
xs.Set(99, 0)
// throws: Array.Set: Can't set index 99 to value '0' in array<Int32> with 89 items.
If you want to use bracket notation xs.[i] with these descriptive errors, use the DebugIdx member:
let item = xs.DebugIdx.[99] // throws with same descriptive message
xs.DebugIdx.[99] <- 0 // setter also has descriptive errors
Positional access
Convenient properties for accessing items at common positions:
let arr = [| "a"; "b"; "c"; "d"; "e" |]
arr.First // "a"
arr.Second // "b"
arr.Third // "c"
arr.Last // "e"
arr.SecondLast // "d"
arr.ThirdLast // "c"
// These are settable too
arr.Last <- "z" // arr is now [| "a"; "b"; "c"; "d"; "z" |]
// For single-element arrays
let single = [| 42 |]
single.FirstAndOnly // 42
Or via module functions for use in pipelines:
[| 10; 20; 30 |] |> Array.first // 10
[| 10; 20; 30 |] |> Array.last // 30
[| 10; 20; 30 |] |> Array.secondLast // 20
Negative indexing (Python-style)
Use negative indices where -1 is the last item, -2 the second last, and so on:
let arr = [| "a"; "b"; "c"; "d"; "e" |]
arr.GetNeg(-1) // "e" (last)
arr.GetNeg(-2) // "d" (second last)
arr.SetNeg(-1, "z")
// Module functions
arr |> Array.getNeg -1 // "e"
arr |> Array.setNeg -2 "x"
Looped indexing
Treats the array as circular, wrapping around in both directions:
let arr = [| "a"; "b"; "c" |]
arr.GetLooped(3) // "a" (wraps around)
arr.GetLooped(4) // "b"
arr.GetLooped(-1) // "c" (wraps backward)
arr.GetLooped(-4) // "c"
Slicing with negative indices
Unlike the built-in a.[1..3] slice syntax, Slice supports negative indices:
let arr = [| 0; 1; 2; 3; 4; 5; 6; 7; 8; 9 |]
arr.Slice(2, 5) // [| 2; 3; 4; 5 |] (inclusive on both ends)
arr.Slice(0, -1) // [| 0; 1; .. ; 9 |] (full copy, -1 = last)
arr.Slice(1, -2) // [| 1; 2; .. ; 8 |] (skip first and last)
arr.Slice(-3, -1) // [| 7; 8; 9 |] (last 3 items)
// Module function
arr |> Array.slice 1 -2 // [| 1; 2; .. ; 8 |]
Trimming
Remove items from the start and end:
let arr = [| 0; 1; 2; 3; 4; 5 |]
arr |> Array.trim 1 1 // [| 1; 2; 3; 4 |] (trim 1 from each end)
arr |> Array.trim 2 0 // [| 2; 3; 4; 5 |] (trim 2 from start)
Rotating
Shift elements circularly:
let arr = [| 1; 2; 3; 4; 5 |]
arr |> Array.rotate 1 // [| 2; 3; 4; 5; 1 |] (rotate up by 1)
arr |> Array.rotate -1 // [| 5; 1; 2; 3; 4 |] (rotate down by 1)
arr |> Array.rotate 2 // [| 3; 4; 5; 1; 2 |] (rotate up by 2)
// Rotate until a condition is met
[| 3; 1; 4; 1; 5 |] |> Array.rotateUpTill (fun x -> x = 5)
// [| 5; 3; 1; 4; 1 |]
Windowed pairs and triples
Iterate over consecutive elements:
let arr = [| "a"; "b"; "c"; "d" |]
// Consecutive pairs (not looped, result is 1 shorter)
arr |> Array.windowed2
// seq { ("a","b"); ("b","c"); ("c","d") }
// Looped pairs (includes wrap-around, same length as input)
arr |> Array.thisNext
// seq { ("a","b"); ("b","c"); ("c","d"); ("d","a") }
arr |> Array.prevThis
// seq { ("d","a"); ("a","b"); ("b","c"); ("c","d") }
// Consecutive triples (not looped, result is 2 shorter)
arr |> Array.windowed3
// seq { ("a","b","c"); ("b","c","d") }
// Looped triples (same length as input)
arr |> Array.prevThisNext
// seq { ("d","a","b"); ("a","b","c"); ("b","c","d"); ("c","d","a") }
With indices:
let arr = [| 10; 20; 30; 40 |]
arr |> Array.windowed2i
// seq { (0, 10, 20); (1, 20, 30); (2, 30, 40) }
arr |> Array.iThisNext
// seq { (0, 10, 20); (1, 20, 30); (2, 30, 40); (3, 40, 10) }
Min and max (top 2 and top 3)
Find the smallest or largest elements:
let arr = [| 5; 1; 9; 3; 7 |]
Array.min2 arr // (1, 3) smallest and second smallest
Array.max2 arr // (9, 7) biggest and second biggest
Array.min3 arr // (1, 3, 5)
Array.max3 arr // (9, 7, 5)
// By projection
let words = [| "hi"; "hello"; "hey" |]
Array.min2By String.length words // ("hi", "hey")
Array.max2By String.length words // ("hello", "hey")
// Get indices instead of values
Array.min2IndicesBy String.length words // (0, 2)
Array.max2IndicesBy String.length words // (1, 2)
Finding duplicates
Array.duplicates [| 1; 2; 3; 2; 4; 3 |]
// [| 2; 3 |] (each duplicate reported once, ordered by first occurrence)
Array.duplicatesBy String.length [| "hi"; "hey"; "go"; "bye" |]
// [| "hi"; "hey" |] (length 2 and length 3 both have duplicates)
Searching
let arr = [| 10; 20; 30; 40; 50 |]
Array.findValue 30 0 4 arr // 2 (found at index 2)
Array.findValue 99 0 4 arr // -1 (not found)
Array.findLastValue 30 0 4 arr // 2 (search from end)
// Search for sub-array pattern
let hay = [| 1; 2; 3; 4; 5; 3; 4 |]
Array.findArray [| 3; 4 |] 0 6 hay // 2 (first match)
Array.findLastArray [| 3; 4 |] 0 6 hay // 5 (last match)
Array checks
let arr = [| 1; 2; 3 |]
arr.IsEmpty // false
arr.IsNotEmpty // true
arr.HasItems // true (same as IsNotEmpty)
arr.IsSingleton // false
// Module functions
arr |> Array.isNotEmpty // true
arr |> Array.isSingleton // false
arr |> Array.hasItems 3 // true (exactly 3 items)
arr |> Array.hasMinimumItems 2 // true (at least 2 items)
arr |> Array.hasMaximumItems 5 // true (at most 5 items)
arr |> Array.count // 3
arr |> Array.countIf (fun x -> x > 1) // 2
Validation
Chainable validation methods for defensive programming:
let result =
someArray
|> Array.failIfEmpty "Input array must not be empty"
|> Array.failIfLessThan 3 "Need at least 3 elements"
|> Array.map doSomething
// Extension member versions
someArray
.FailIfEmpty("Input must not be empty")
.FailIfLessThan(3, "Need at least 3 elements")
Swapping
let arr = [| "a"; "b"; "c"; "d" |]
Array.swap 0 3 arr
// arr is now [| "d"; "b"; "c"; "a" |]
String representation
let arr = [| 1; 2; 3; 4; 5; 6; 7 |]
arr.asString
// "array<Int32> with 7 items:
// 0: 1
// 1: 2
// 2: 3
// 3: 4
// 4: 5
// ..."
arr.ToString(3) // show only first 3 entries
// "array<Int32> with 7 items:
// 0: 1
// 1: 2
// 2: 3
// ..."
Use of AI and LLMs
All core function are are written by hand to ensure performance and correctness.
However, AI tools have been used for code review, typo and grammar checking in documentation
and to generate not all but many of the tests.
Tests
All Tests run in both javascript and dotnet. Successful Fable compilation to typescript is verified too. Go to the tests folder:
cd Tests
For testing with .NET using Expecto:
dotnet run
for JS testing with Fable.Mocha and TS verification:
npm test
License
Changelog
see CHANGELOG.md
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
| .NET Framework | net472 is compatible. net48 was computed. net481 was computed. |
-
.NETFramework 4.7.2
- FSharp.Core (>= 6.0.7)
-
net6.0
- FSharp.Core (>= 6.0.7)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
### Changed
- allow Array.asResizeArray only on reference types