Basic operations on List
. #
We define
- basic operations on
List
, - simp lemmas for applying the operations on
.nil
and.cons
arguments (in the cases where the right hand side is simple to state; otherwise these are deferred toInit.Data.List.Lemmas
), - the minimal lemmas which are required for setting up
Init.Data.Array.Basic
.
In Init.Data.List.Impl
we give tail-recursive versions of these operations
along with @[csimp]
lemmas,
In Init.Data.List.Lemmas
we develop the full API for these functions.
Recall that length
, get
, set
, foldl
, and concat
have already been defined in Init.Prelude
.
The operations are organized as follow:
- Equality:
beq
,isEqv
. - Lexicographic ordering:
lt
,le
, and instances. - Head and tail operators:
head
,head?
,headD?
,tail
,tail?
,tailD
. - Basic operations:
map
,filter
,filterMap
,foldr
,append
,flatten
,pure
,flatMap
,replicate
, andreverse
. - Additional functions defined in terms of these:
leftpad
,rightPad
, andreduceOption
. - Operations using indexes:
mapIdx
. - List membership:
isEmpty
,elem
,contains
,mem
(and the∈
notation), and decidability for predicates quantifying over membership in aList
. - Sublists:
take
,drop
,takeWhile
,dropWhile
,partition
,dropLast
,isPrefixOf
,isPrefixOf?
,isSuffixOf
,isSuffixOf?
,Subset
,Sublist
,rotateLeft
androtateRight
. - Manipulating elements:
replace
,modify
,insert
,insertIdx
,erase
,eraseP
,eraseIdx
. - Finding elements:
find?
,findSome?
,findIdx
,indexOf
,findIdx?
,indexOf?
,countP
,count
, andlookup
. - Logic:
any
,all
,or
, andand
. - Zippers:
zipWith
,zip
,zipWithAll
, andunzip
. - Ranges and enumeration:
range
,iota
,enumFrom
, andenum
. - Minima and maxima:
min?
andmax?
. - Other functions:
intersperse
,intercalate
,eraseDups
,eraseReps
,span
,splitBy
,removeAll
(currently these functions are mostly only used in meta code, and do not have API suitable for verification).
Further operations are defined in Init.Data.List.BasicAux
(because they use Array
in their implementations), namely:
- Variant getters:
get!
,get?
,getD
,getLast
,getLast!
,getLast?
, andgetLastD
. - Head and tail:
head!
,tail!
. - Other operations on sublists:
partitionMap
,rotateLeft
, androtateRight
.
Preliminaries from Init.Prelude
#
length #
set #
foldl #
concat #
Equality #
Equations
- List.instBEq = { beq := List.beq }
O(min |as| |bs|)
. Returns true if as
and bs
have the same length,
and they are pairwise related by eqv
.
Equations
Instances For
Lexicographic ordering #
Lexicographic ordering for lists.
- nil
{α : Type u}
{r : α → α → Prop}
{a : α}
{l : List α}
: List.Lex r [] (a :: l)
[]
is the smallest element in the order. - cons
{α : Type u}
{r : α → α → Prop}
{a : α}
{l₁ l₂ : List α}
(h : List.Lex r l₁ l₂)
: List.Lex r (a :: l₁) (a :: l₂)
If
a
is indistinguishable fromb
andas < bs
, thena::as < b::bs
. - rel
{α : Type u}
{r : α → α → Prop}
{a₁ : α}
{l₁ : List α}
{a₂ : α}
{l₂ : List α}
(h : r a₁ a₂)
: List.Lex r (a₁ :: l₁) (a₂ :: l₂)
If
a < b
thena::as < b::bs
.
Instances For
Equations
- One or more equations did not get rendered due to their size.
- List.decidableLex r [] [] = isFalse ⋯
- List.decidableLex r [] (head :: tail) = isTrue ⋯
- List.decidableLex r (head :: tail) [] = isFalse ⋯
Decidability of lexicographic ordering.
Equations
- l₁.decidableLT l₂ = List.decidableLex (fun (x1 x2 : α) => x1 < x2) l₁ l₂
Decidability of lexicographic ordering.
Equations
Instances For
Equations
- l₁.decidableLE l₂ = inferInstanceAs (Decidable ¬l₂ < l₁)
Alternative getters #
get? #
Returns the i
-th element in the list (zero-based).
If the index is out of bounds (i ≥ as.length
), this function returns none
.
Also see get
, getD
and get!
.
Instances For
Deprecated alias for ext_get?
. The preferred extensionality theorem is now ext_getElem?
.
Equations
Instances For
getD #
getLast #
getLast? #
getLastD #
Head and tail #
head #
head? #
headD #
tail #
tail? #
tailD #
Basic List
operations. #
We define the basic functional programming operations on List
:
map
, filter
, filterMap
, foldr
, append
, flatten
, pure
, bind
, replicate
, and reverse
.
map #
filter #
O(|l|)
. filter f l
returns the list of elements in l
for which f
returns true.
filter (· > 2) [1, 2, 5, 2, 7, 7] = [5, 7, 7]
Equations
- List.filter p [] = []
- List.filter p (a :: as) = match p a with | true => a :: List.filter p as | false => List.filter p as
Instances For
filterMap #
O(|l|)
. filterMap f l
takes a function f : α → Option β
and applies it to each element of l
;
the resulting non-none
values are collected to form the output list.
filterMap
(fun x => if x > 2 then some (2 * x) else none)
[1, 2, 5, 2, 7, 7]
= [10, 14, 14]
Equations
- List.filterMap f [] = []
- List.filterMap f (a :: as) = match f a with | none => List.filterMap f as | some b => b :: List.filterMap f as
Instances For
foldr #
O(|l|)
. Applies function f
to all of the elements of the list, from right to left.
foldr f init [a, b, c] = f a <| f b <| f c <| init
Equations
- List.foldr f init [] = init
- List.foldr f init (a :: as) = f a (List.foldr f init as)
Instances For
reverse #
Auxiliary for List.reverse
. List.reverseAux l r = l.reverse ++ r
, but it is defined directly.
Instances For
O(|as|)
. Reverse of a list:
[1, 2, 3, 4].reverse = [4, 3, 2, 1]
Note that because of the "functional but in place" optimization implemented by Lean's compiler, this function works without any allocations provided that the input list is unshared: it simply walks the linked list and reverses all the node pointers.
Equations
- as.reverse = as.reverseAux []
Instances For
append #
Tail-recursive version of List.append
.
Most of the tail-recursive implementations are in Init.Data.List.Impl
,
but appendTR
must be set up immediately,
because otherwise Append (List α)
instance below will not use it.
Equations
- as.appendTR bs = as.reverse.reverseAux bs
Instances For
Equations
- List.instAppend = { append := List.append }
flatten #
singleton #
flatMap #
flatMap xs f
applies f
to each element of xs
to get a list of lists, and then concatenates them all together.
Instances For
Equations
Instances For
Equations
Instances For
Equations
Instances For
Equations
Instances For
replicate #
replicate n a
is n
copies of a
:
replicate 5 a = [a, a, a, a, a]
Equations
- List.replicate 0 x✝ = []
- List.replicate n.succ x✝ = x✝ :: List.replicate n x✝
Instances For
Additional functions #
leftpad and rightpad #
Pads l : List α
on the left with repeated occurrences of a : α
until it is of length n
.
If l
is initially larger than n
, just return l
.
Equations
- List.leftpad n a l = List.replicate (n - l.length) a ++ l
Instances For
Pads l : List α
on the right with repeated occurrences of a : α
until it is of length n
.
If l
is initially larger than n
, just return l
.
Equations
- List.rightpad n a l = l ++ List.replicate (n - l.length) a
Instances For
reduceOption #
Drop none
s from a list, and replace each remaining some a
with a
.
Equations
Instances For
List membership #
EmptyCollection #
Equations
- List.instEmptyCollection = { emptyCollection := [] }
isEmpty #
elem #
contains #
Mem #
a ∈ l
is a predicate which asserts that a
is in the list l
.
Unlike elem
, this uses =
instead of ==
and is suited for mathematical reasoning.
a ∈ [x, y, z] ↔ a = x ∨ a = y ∨ a = z
- head
{α : Type u}
{a : α}
(as : List α)
: List.Mem a (a :: as)
The head of a list is a member:
a ∈ a :: as
. - tail
{α : Type u}
{a : α}
(b : α)
{as : List α}
: List.Mem a as → List.Mem a (b :: as)
A member of the tail of a list is a member of the list:
a ∈ l → a ∈ b :: l
.
Instances For
Equations
- List.instMembership = { mem := fun (l : List α) (a : α) => List.Mem a l }
Equations
Equations
- List.decidableBEx p [] = isFalse ⋯
- List.decidableBEx p (a :: as) = if h₁ : p a then isTrue ⋯ else match List.decidableBEx p as with | isTrue h₂ => isTrue ⋯ | isFalse h₂ => isFalse ⋯
Equations
- List.decidableBAll p [] = isTrue ⋯
- List.decidableBAll p (a :: as) = if h₁ : p a then match List.decidableBAll p as with | isTrue h₂ => isTrue ⋯ | isFalse h₂ => isFalse ⋯ else isFalse ⋯
Sublists #
take #
drop #
takeWhile #
O(|xs|)
. Returns the longest initial segment of xs
for which p
returns true.
Equations
- List.takeWhile p [] = []
- List.takeWhile p (a :: as) = match p a with | true => a :: List.takeWhile p as | false => []
Instances For
dropWhile #
O(|l|)
. dropWhile p l
removes elements from the list until it finds the first element
for which p
returns false; this element and everything after it is returned.
dropWhile (· < 4) [1, 3, 2, 4, 2, 7, 4] = [4, 2, 7, 4]
Equations
- List.dropWhile p [] = []
- List.dropWhile p (a :: as) = match p a with | true => List.dropWhile p as | false => a :: as
Instances For
partition #
O(|l|)
. partition p l
calls p
on each element of l
, partitioning the list into two lists
(l_true, l_false)
where l_true
has the elements where p
was true
and l_false
has the elements where p
is false.
partition p l = (filter p l, filter (not ∘ p) l)
, but it is slightly more efficient
since it only has to do one pass over the list.
partition (· > 2) [1, 2, 5, 2, 7, 7] = ([5, 7, 7], [1, 2, 2])
Equations
- List.partition p as = List.partition.loop p as ([], [])
Instances For
Equations
- List.partition.loop p [] (bs, cs) = (bs.reverse, cs.reverse)
- List.partition.loop p (a :: as) (bs, cs) = match p a with | true => List.partition.loop p as (a :: bs, cs) | false => List.partition.loop p as (bs, a :: cs)
Instances For
dropLast #
Subset #
Equations
- List.instHasSubset = { Subset := List.Subset }
Equations
- x✝¹.instDecidableRelSubsetOfDecidableEq x✝ = List.decidableBAll (Membership.mem x✝) x✝¹
Sublist and isSublist #
l₁ <+ l₂
, or Sublist l₁ l₂
, says that l₁
is a (non-contiguous) subsequence of l₂
.
- slnil
{α : Type u_1}
: [].Sublist []
the base case:
[]
is a sublist of[]
- cons
{α : Type u_1}
{l₁ l₂ : List α}
(a : α)
: l₁.Sublist l₂ → l₁.Sublist (a :: l₂)
If
l₁
is a subsequence ofl₂
, then it is also a subsequence ofa :: l₂
. - cons₂
{α : Type u_1}
{l₁ l₂ : List α}
(a : α)
: l₁.Sublist l₂ → (a :: l₁).Sublist (a :: l₂)
If
l₁
is a subsequence ofl₂
, thena :: l₁
is a subsequence ofa :: l₂
.
Instances For
l₁ <+ l₂
, or Sublist l₁ l₂
, says that l₁
is a (non-contiguous) subsequence of l₂
.
Equations
- List.«term_<+_» = Lean.ParserDescr.trailingNode `List.«term_<+_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <+ ") (Lean.ParserDescr.cat `term 51))
Instances For
True if the first list is a potentially non-contiguous sub-sequence of the second list.
Equations
Instances For
IsPrefix / isPrefixOf / isPrefixOf? #
IsPrefix l₁ l₂
, or l₁ <+: l₂
, means that l₁
is a prefix of l₂
,
that is, l₂
has the form l₁ ++ t
for some t
.
Equations
- List.«term_<+:_» = Lean.ParserDescr.trailingNode `List.«term_<+:_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <+: ") (Lean.ParserDescr.cat `term 51))
Instances For
isPrefixOf l₁ l₂
returns true
Iff l₁
is a prefix of l₂
.
That is, there exists a t
such that l₂ == l₁ ++ t
.
Equations
Instances For
IsSuffix / isSuffixOf / isSuffixOf? #
isSuffixOf l₁ l₂
returns true
Iff l₁
is a suffix of l₂
.
That is, there exists a t
such that l₂ == t ++ l₁
.
Equations
- l₁.isSuffixOf l₂ = l₁.reverse.isPrefixOf l₂.reverse
Instances For
isSuffixOf? l₁ l₂
returns some t
when l₂ == t ++ l₁
.
Equations
- l₁.isSuffixOf? l₂ = Option.map List.reverse (l₁.reverse.isPrefixOf? l₂.reverse)
Instances For
IsSuffix l₁ l₂
, or l₁ <:+ l₂
, means that l₁
is a suffix of l₂
,
that is, l₂
has the form t ++ l₁
for some t
.
Equations
- List.«term_<:+_» = Lean.ParserDescr.trailingNode `List.«term_<:+_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <:+ ") (Lean.ParserDescr.cat `term 51))
Instances For
IsInfix #
IsInfix l₁ l₂
, or l₁ <:+: l₂
, means that l₁
is a contiguous
substring of l₂
, that is, l₂
has the form s ++ l₁ ++ t
for some s, t
.
Equations
- List.«term_<:+:_» = Lean.ParserDescr.trailingNode `List.«term_<:+:_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <:+: ") (Lean.ParserDescr.cat `term 51))
Instances For
splitAt #
Split a list at an index.
splitAt 2 [a, b, c] = ([a, b], [c])
Equations
- List.splitAt n l = List.splitAt.go l l n []
Instances For
Auxiliary for splitAt
:
splitAt.go l xs n acc = (acc.reverse ++ take n xs, drop n xs)
if n < xs.length
,
and (l, [])
otherwise.
Equations
- List.splitAt.go l [] x✝¹ x✝ = (l, [])
- List.splitAt.go l (x_3 :: xs) n.succ x✝ = List.splitAt.go l xs n (x_3 :: x✝)
- List.splitAt.go l x✝² x✝¹ x✝ = (x✝.reverse, x✝²)
Instances For
rotateLeft #
O(n)
. Rotates the elements of xs
to the left such that the element at
xs[i]
rotates to xs[(i - n) % l.length]
.
rotateLeft [1, 2, 3, 4, 5] 3 = [4, 5, 1, 2, 3]
rotateLeft [1, 2, 3, 4, 5] 5 = [1, 2, 3, 4, 5]
rotateLeft [1, 2, 3, 4, 5] = [2, 3, 4, 5, 1]
Equations
Instances For
rotateRight #
O(n)
. Rotates the elements of xs
to the right such that the element at
xs[i]
rotates to xs[(i + n) % l.length]
.
rotateRight [1, 2, 3, 4, 5] 3 = [3, 4, 5, 1, 2]
rotateRight [1, 2, 3, 4, 5] 5 = [1, 2, 3, 4, 5]
rotateRight [1, 2, 3, 4, 5] = [5, 1, 2, 3, 4]
Equations
Instances For
Pairwise, Nodup #
Pairwise R l
means that all the elements with earlier indexes are
R
-related to all the elements with later indexes.
Pairwise R [1, 2, 3] ↔ R 1 2 ∧ R 1 3 ∧ R 2 3
For example if R = (·≠·)
then it asserts l
has no duplicates,
and if R = (·<·)
then it asserts that l
is (strictly) sorted.
- nil
{α : Type u}
{R : α → α → Prop}
: List.Pairwise R []
All elements of the empty list are vacuously pairwise related.
- cons {α : Type u} {R : α → α → Prop} {a : α} {l : List α} : (∀ (a' : α), a' ∈ l → R a a') → List.Pairwise R l → List.Pairwise R (a :: l)
Instances For
Nodup l
means that l
has no duplicates, that is, any element appears at most
once in the List. It is defined as Pairwise (≠)
.
Equations
- List.Nodup = List.Pairwise fun (x1 x2 : α) => x1 ≠ x2
Instances For
Manipulating elements #
replace #
modify #
Apply a function to the nth tail of l
. Returns the input without
using f
if the index is larger than the length of the List.
modifyTailIdx f 2 [a, b, c] = [a, b] ++ f [c]
Equations
- List.modifyTailIdx f 0 x✝ = f x✝
- List.modifyTailIdx f n.succ [] = []
- List.modifyTailIdx f n.succ (a :: as) = a :: List.modifyTailIdx f n as
Instances For
Apply f
to the head of the list, if it exists.
Equations
- List.modifyHead f [] = []
- List.modifyHead f (a :: as) = f a :: as
Instances For
Apply f
to the nth element of the list, if it exists, replacing that element with the result.
Equations
Instances For
insert #
insertIdx n a l
inserts a
into the list l
after the first n
elements of l
insertIdx 2 1 [1, 2, 3, 4] = [1, 2, 1, 3, 4]
Equations
- List.insertIdx n a = List.modifyTailIdx (List.cons a) n
Instances For
erase #
eraseP p l
removes the first element of l
satisfying the predicate p
.
Equations
- List.eraseP p [] = []
- List.eraseP p (a :: as) = bif p a then as else a :: List.eraseP p as
Instances For
eraseIdx #
Finding elements
find? #
O(|l|)
. find? p l
returns the first element for which p
returns true,
or none
if no such element is found.
Equations
- List.find? p [] = none
- List.find? p (a :: as) = match p a with | true => some a | false => List.find? p as
Instances For
findSome? #
O(|l|)
. findSome? f l
applies f
to each element of l
, and returns the first non-none
result.
findSome? (fun x => if x < 5 then some (10 * x) else none) [7, 6, 5, 8, 1, 2, 6] = some 10
Equations
- List.findSome? f [] = none
- List.findSome? f (a :: as) = match f a with | some b => some b | none => List.findSome? f as
Instances For
findIdx #
Returns the index of the first element satisfying p
, or the length of the list otherwise.
Equations
- List.findIdx p l = List.findIdx.go p l 0
Instances For
Auxiliary for findIdx
: findIdx.go p l n = findIdx p l + n
Equations
- List.findIdx.go p [] x✝ = x✝
- List.findIdx.go p (a :: l) x✝ = bif p a then x✝ else List.findIdx.go p l (x✝ + 1)
Instances For
indexOf #
Returns the index of the first element equal to a
, or the length of the list otherwise.
Equations
- List.indexOf a = List.findIdx fun (x : α) => x == a
Instances For
findIdx? #
Return the index of the first occurrence of an element satisfying p
.
Equations
- List.findIdx? p [] x✝ = none
- List.findIdx? p (a :: l) x✝ = if p a = true then some x✝ else List.findIdx? p l (x✝ + 1)
Instances For
indexOf? #
Return the index of the first occurrence of a
in the list.
Equations
- List.indexOf? a a✝ = List.findIdx? (fun (x : α) => x == a) a✝
Instances For
countP #
countP p l
is the number of elements of l
that satisfy p
.
Equations
- List.countP p l = List.countP.go p l 0
Instances For
Auxiliary for countP
: countP.go p l acc = countP p l + acc
.
Equations
- List.countP.go p [] x✝ = x✝
- List.countP.go p (a :: l) x✝ = bif p a then List.countP.go p l (x✝ + 1) else List.countP.go p l x✝
Instances For
count #
count a l
is the number of occurrences of a
in l
.
Equations
- List.count a = List.countP fun (x : α) => x == a
Instances For
lookup #
O(|l|)
. lookup a l
treats l : List (α × β)
like an association list,
and returns the first β
value corresponding to an α
value in the list equal to a
.
Equations
- List.lookup x✝ [] = none
- List.lookup x✝ ((k, b) :: es) = match x✝ == k with | true => some b | false => List.lookup x✝ es
Instances For
Permutations #
Perm #
Perm l₁ l₂
or l₁ ~ l₂
asserts that l₁
and l₂
are permutations
of each other. This is defined by induction using pairwise swaps.
- nil
{α : Type u}
: [].Perm []
[] ~ []
- cons
{α : Type u}
(x : α)
{l₁ l₂ : List α}
: l₁.Perm l₂ → (x :: l₁).Perm (x :: l₂)
l₁ ~ l₂ → x::l₁ ~ x::l₂
- swap
{α : Type u}
(x y : α)
(l : List α)
: (y :: x :: l).Perm (x :: y :: l)
x::y::l ~ y::x::l
- trans
{α : Type u}
{l₁ l₂ l₃ : List α}
: l₁.Perm l₂ → l₂.Perm l₃ → l₁.Perm l₃
Perm
is transitive.
Instances For
Perm l₁ l₂
or l₁ ~ l₂
asserts that l₁
and l₂
are permutations
of each other. This is defined by induction using pairwise swaps.
Equations
- List.«term_~_» = Lean.ParserDescr.trailingNode `List.«term_~_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " ~ ") (Lean.ParserDescr.cat `term 51))
Instances For
isPerm #
Logical operations #
any #
all #
or #
and #
Zippers #
zipWith #
O(min |xs| |ys|)
. Applies f
to the two lists in parallel, stopping at the shorter list.
zipWith f [x₁, x₂, x₃] [y₁, y₂, y₃, y₄] = [f x₁ y₁, f x₂ y₂, f x₃ y₃]
Equations
- List.zipWith f (x_2 :: xs) (y :: ys) = f x_2 y :: List.zipWith f xs ys
- List.zipWith f x✝¹ x✝ = []
Instances For
zip #
O(min |xs| |ys|)
. Combines the two lists into a list of pairs, with one element from each list.
The longer list is truncated to match the shorter list.
zip [x₁, x₂, x₃] [y₁, y₂, y₃, y₄] = [(x₁, y₁), (x₂, y₂), (x₃, y₃)]
Equations
Instances For
zipWithAll #
O(max |xs| |ys|)
.
Version of List.zipWith
that continues to the end of both lists,
passing none
to one argument once the shorter list has run out.
Equations
- List.zipWithAll f [] x✝ = List.map (fun (b : β) => f none (some b)) x✝
- List.zipWithAll f (a :: as) [] = List.map (fun (a : α) => f (some a) none) (a :: as)
- List.zipWithAll f (a :: as) (b :: bs) = f (some a) (some b) :: List.zipWithAll f as bs
Instances For
unzip #
O(|l|)
. Separates a list of pairs into two lists containing the first components and second components.
unzip [(x₁, y₁), (x₂, y₂), (x₃, y₃)] = ([x₁, x₂, x₃], [y₁, y₂, y₃])
Equations
Instances For
Ranges and enumeration #
range #
O(n)
. range n
is the numbers from 0
to n
exclusive, in increasing order.
range 5 = [0, 1, 2, 3, 4]
Equations
- List.range n = List.range.loop n []
Instances For
Equations
- List.range.loop 0 x✝ = x✝
- List.range.loop n.succ x✝ = List.range.loop n (n :: x✝)
Instances For
range' #
range' start len step
is the list of numbers [start, start+step, ..., start+(len-1)*step]
.
It is intended mainly for proving properties of range
and iota
.
Equations
- List.range' x✝¹ 0 x✝ = []
- List.range' x✝¹ n.succ x✝ = x✝¹ :: List.range' (x✝¹ + x✝) n x✝
Instances For
iota #
enumFrom #
O(|l|)
. enumFrom n l
is like enum
but it allows you to specify the initial index.
enumFrom 5 [a, b, c] = [(5, a), (6, b), (7, c)]
Equations
- List.enumFrom x✝ [] = []
- List.enumFrom x✝ (x_2 :: xs) = (x✝, x_2) :: List.enumFrom (x✝ + 1) xs
Instances For
enum #
Minima and maxima #
min? #
Returns the smallest element of the list, if it is not empty.
Equations
Instances For
max? #
Returns the largest element of the list, if it is not empty.
Equations
Instances For
Other list operations #
The functions are currently mostly used in meta code, and do not have sufficient API developed for verification work.
intersperse #
O(|l|)
. intersperse sep l
alternates sep
and the elements of l
:
intersperse sep [] = []
intersperse sep [a] = [a]
intersperse sep [a, b] = [a, sep, b]
intersperse sep [a, b, c] = [a, sep, b, sep, c]
Equations
- List.intersperse sep [] = []
- List.intersperse sep [head] = [head]
- List.intersperse sep (a :: as) = a :: sep :: List.intersperse sep as
Instances For
intercalate #
O(|xs|)
. intercalate sep xs
alternates sep
and the elements of xs
:
intercalate sep [] = []
intercalate sep [a] = a
intercalate sep [a, b] = a ++ sep ++ b
intercalate sep [a, b, c] = a ++ sep ++ b ++ sep ++ c
Equations
- sep.intercalate xs = (List.intersperse sep xs).flatten
Instances For
eraseDups #
O(|l|^2)
. Erase duplicated elements in the list.
Keeps the first occurrence of duplicated elements.
eraseDups [1, 3, 2, 2, 3, 5] = [1, 3, 2, 5]
Equations
- as.eraseDups = List.eraseDups.loop as []
Instances For
Equations
- List.eraseDups.loop [] x✝ = x✝.reverse
- List.eraseDups.loop (a :: l) x✝ = match List.elem a x✝ with | true => List.eraseDups.loop l x✝ | false => List.eraseDups.loop l (a :: x✝)
Instances For
eraseReps #
Equations
- List.eraseReps.loop x✝¹ [] x✝ = (x✝¹ :: x✝).reverse
- List.eraseReps.loop x✝¹ (a' :: as) x✝ = match x✝¹ == a' with | true => List.eraseReps.loop x✝¹ as x✝ | false => List.eraseReps.loop a' as (x✝¹ :: x✝)
Instances For
span #
O(|l|)
. span p l
splits the list l
into two parts, where the first part
contains the longest initial segment for which p
returns true
and the second part is everything else.
span (· > 5) [6, 8, 9, 5, 2, 9] = ([6, 8, 9], [5, 2, 9])
span (· > 10) [6, 8, 9, 5, 2, 9] = ([], [6, 8, 9, 5, 2, 9])
Equations
- List.span p as = List.span.loop p as []
Instances For
Equations
- List.span.loop p [] x✝ = (x✝.reverse, [])
- List.span.loop p (a :: l) x✝ = match p a with | true => List.span.loop p l (a :: x✝) | false => (x✝.reverse, a :: l)
Instances For
splitBy #
O(|l|)
. splitBy R l
splits l
into chains of elements
such that adjacent elements are related by R
.
splitBy (·==·) [1, 1, 2, 2, 2, 3, 2] = [[1, 1], [2, 2, 2], [3], [2]]
splitBy (·<·) [1, 2, 5, 4, 5, 1, 4] = [[1, 2, 5], [4, 5], [1, 4]]
Equations
- List.splitBy R [] = []
- List.splitBy R (a :: as) = List.splitBy.loop R as a [] []
Instances For
The arguments of splitBy.loop l ag g gs
represent the following:
l : List α
are the elements which we still need to split.ag : α
is the previous element for which a comparison was performed.g : List α
is the group currently being assembled, in reverse order.gs : List (List α)
is all of the groups that have been completed, in reverse order.
Equations
- List.splitBy.loop R (a :: as) x✝² x✝¹ x✝ = match R x✝² a with | true => List.splitBy.loop R as a (x✝² :: x✝¹) x✝ | false => List.splitBy.loop R as a [] ((x✝² :: x✝¹).reverse :: x✝)
- List.splitBy.loop R [] x✝² x✝¹ x✝ = ((x✝² :: x✝¹).reverse :: x✝).reverse
Instances For
O(|l|)
. splitBy R l
splits l
into chains of elements
such that adjacent elements are related by R
.
splitBy (·==·) [1, 1, 2, 2, 2, 3, 2] = [[1, 1], [2, 2, 2], [3], [2]]
splitBy (·<·) [1, 2, 5, 4, 5, 1, 4] = [[1, 2, 5], [4, 5], [1, 4]]
Equations
Instances For
removeAll #
O(|xs|)
. Computes the "set difference" of lists,
by filtering out all elements of xs
which are also in ys
.
removeAll [1, 1, 5, 1, 2, 4, 5] [1, 2, 2] = [5, 4, 5]
Equations
- xs.removeAll ys = List.filter (fun (x : α) => !List.elem x ys) xs
Instances For
Runtime re-implementations using @[csimp]
#
More of these re-implementations are provided in Init/Data/List/Impl.lean
.
They can not be here, because the remaining ones required Array
for their implementation.
This leaves a dangerous situation: if you import this file, but not Init/Data/List/Impl.lean
,
then at runtime you will get non tail-recursive versions.
length #
map #
Tail-recursive version of List.map
.
Equations
- List.mapTR f as = List.mapTR.loop f as []
Instances For
Equations
- List.mapTR.loop f [] x✝ = x✝.reverse
- List.mapTR.loop f (a :: as) x✝ = List.mapTR.loop f as (f a :: x✝)
Instances For
filter #
Tail-recursive version of List.filter
.
Equations
- List.filterTR p as = List.filterTR.loop p as []
Instances For
Equations
- List.filterTR.loop p [] x✝ = x✝.reverse
- List.filterTR.loop p (a :: l) x✝ = match p a with | true => List.filterTR.loop p l (a :: x✝) | false => List.filterTR.loop p l x✝
Instances For
replicate #
Tail-recursive version of List.replicate
.
Equations
- List.replicateTR n a = List.replicateTR.loop a n []
Instances For
Equations
- List.replicateTR.loop a 0 x✝ = x✝
- List.replicateTR.loop a n.succ x✝ = List.replicateTR.loop a n (a :: x✝)
Instances For
Additional functions #
leftpad #
Optimized version of leftpad
.
Equations
- List.leftpadTR n a l = List.replicateTR.loop a (n - l.length) l
Instances For
Zippers #
unzip #
Ranges and enumeration #
range' #
Optimized version of range'
.
Equations
- List.range'TR s n step = List.range'TR.go step n (s + step * n) []
Instances For
Auxiliary for range'TR
: range'TR.go n e = [e-n, ..., e-1] ++ acc
.
Equations
- List.range'TR.go step 0 x✝¹ x✝ = x✝
- List.range'TR.go step n.succ x✝¹ x✝ = List.range'TR.go step n (x✝¹ - step) ((x✝¹ - step) :: x✝)
Instances For
iota #
Equations
- List.iotaTR.go 0 x✝ = x✝.reverse
- List.iotaTR.go n.succ x✝ = List.iotaTR.go n (n.succ :: x✝)
Instances For
Other list operations #
intersperse #
Tail recursive version of List.intersperse
.
Equations
- List.intersperseTR sep [] = []
- List.intersperseTR sep [x_1] = [x_1]
- List.intersperseTR sep (x_1 :: y :: xs) = x_1 :: sep :: y :: List.foldr (fun (a : α) (r : List α) => sep :: a :: r) [] xs