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
,zipIdx
. - 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 }
Returns true
if as
and bs
have the same length and they are pairwise related by eqv
.
O(min |as| |bs|)
. Short-circuits at the first non-related pair of elements.
Examples:
Lexicographic ordering #
Lexicographic ordering for lists with respect to an ordering of elements.
as
is lexicographically smaller than bs
if
as
is empty andbs
is non-empty, or- both
as
andbs
are non-empty, and the head ofas
is less than the head ofbs
according tor
, or - both
as
andbs
are non-empty, their heads are equal, and the tail ofas
is less than the tail ofbs
.
- nil
{α : Type u}
{r : α → α → Prop}
{a : α}
{l : List α}
: Lex r [] (a :: l)
[]
is the smallest element in the lexicographic order. - rel
{α : Type u}
{r : α → α → Prop}
{a₁ : α}
{l₁ : List α}
{a₂ : α}
{l₂ : List α}
(h : r a₁ a₂)
: Lex r (a₁ :: l₁) (a₂ :: l₂)
If the head of the first list is smaller than the head of the second, then the first list is lexicographically smaller than the second list.
- cons
{α : Type u}
{r : α → α → Prop}
{a : α}
{l₁ l₂ : List α}
(h : Lex r l₁ l₂)
: Lex r (a :: l₁) (a :: l₂)
If two lists have the same head, then their tails determine their lexicographic order. If the tail of the first list is lexicographically smaller than the tail of the second list, then the entire first list is lexicographically smaller than the entire second list.
Instances For
Lexicographic ordering of lists with respect to an ordering on their elements.
as < bs
if
as
is empty andbs
is non-empty, or- both
as
andbs
are non-empty, and the head ofas
is less than the head ofbs
, or - both
as
andbs
are non-empty, their heads are equal, and the tail ofas
is less than the tail ofbs
.
Decidability of lexicographic ordering.
Equations
- l₁.decidableLT l₂ = List.decidableLex (fun (x1 x2 : α) => x1 < x2) l₁ l₂
Decidability of lexicographic ordering.
Equations
Non-strict ordering of lists with respect to a strict ordering of their elements.
as ≤ bs
if ¬ bs < as
.
This relation can be treated as a lexicographic order if the underlying LT α
instance is
well-behaved. In particular, it should be irreflexive, asymmetric, and antisymmetric. These
requirements are precisely formulated in List.cons_le_cons_iff
. If these hold, then as ≤ bs
if
and only if:
as
is empty, or- both
as
andbs
are non-empty, and the head ofas
is less than the head ofbs
, or - both
as
andbs
are non-empty, their heads are equal, and the tail ofas
is less than or equal to the tail ofbs
.
Equations
- l₁.decidableLE l₂ = inferInstanceAs (Decidable ¬l₂ < l₁)
Compares lists lexicographically with respect to a comparison on their elements.
The lexicographic order with respect to lt
is:
Alternative getters #
getLast #
Returns the last element of a non-empty list.
Examples:
getLast? #
Returns the last element in the list, or none
if the list is empty.
Alternatives include List.getLastD
, which takes a fallback value for empty lists, and
List.getLast!
, which panics on empty lists.
Examples:
getLastD #
Returns the last element in the list, or fallback
if the list is empty.
Alternatives include List.getLast?
, which returns an Option
, and List.getLast!
, which panics
on empty lists.
Examples:
Head and tail #
head #
head? #
Returns the first element in the list, if there is one. Returns none
if the list is empty.
Use List.headD
to provide a fallback value for empty lists, or List.head!
to panic on empty
lists.
Examples:
headD #
Returns the first element in the list if there is one, or fallback
if the list is empty.
Use List.head?
to return an Option
, and List.head!
to panic on empty lists.
Examples:
tail #
Drops the first element of a nonempty list, returning the tail. Returns []
when the argument is
empty.
Examples:
tail? #
Drops the first element of a nonempty list, returning the tail. Returns none
when the argument is
empty.
Alternatives include List.tail
, which returns the empty list on failure, List.tailD
, which
returns an explicit fallback value, and List.tail!
, which panics on the empty list.
Examples:
tailD #
Drops the first element of a nonempty list, returning the tail. Returns none
when the argument is
empty.
Alternatives include List.tail
, which returns the empty list on failure, List.tail?
, which
returns an Option
, and List.tail!
, which panics on the empty list.
Examples:
Basic List
operations. #
We define the basic functional programming operations on List
:
map
, filter
, filterMap
, foldr
, append
, flatten
, pure
, bind
, replicate
, and reverse
.
map #
Applies a function to each element of the list, returning the resulting list of values.
O(|l|)
.
Examples:
filter #
Returns the list of elements in l
for which p
returns true
.
O(|l|)
.
Examples:
[1, 2, 5, 2, 7, 7].filter (· > 2) = [5, 7, 7]
[1, 2, 5, 2, 7, 7].filter (fun _ => false) = []
[1, 2, 5, 2, 7, 7].filter (fun _ => true) = [1, 2, 5, 2, 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
filterMap #
Applies a function that returns an Option
to each element of a list, collecting the non-none
values.
O(|l|)
.
Example:
#eval [1, 2, 5, 2, 7, 7].filterMap fun x =>
if x > 2 then some (2 * x) else none
[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
foldr #
Folds a function over a list from the right, accumulating a value starting with init
. The
accumulated value is combined with the each element of the list in reverse order, using f
.
O(|l|)
. Replaced at runtime with List.foldrTR
.
Examples:
[a, b, c].foldr f init = f a (f b (f c init))
[1, 2, 3].foldr (toString · ++ ·) "" = "123"
[1, 2, 3].foldr (s!"({·} {·})") "!" = "(1 (2 (3 !)))"
Equations
- List.foldr f init [] = init
- List.foldr f init (a :: as) = f a (List.foldr f init as)
reverse #
Auxiliary for List.reverse
. List.reverseAux l r = l.reverse ++ r
, but it is defined directly.
Equations
- [].reverseAux x✝ = x✝
- (a :: l).reverseAux x✝ = l.reverseAux (a :: x✝)
Reverses a list.
O(|as|)
.
Because of the “functional but in place” optimization implemented by Lean's compiler, this function does not allocate a new list when its reference to the input list is unshared: it simply walks the linked list and reverses all the node pointers.
Examples:
Equations
- as.reverse = as.reverseAux []
append #
Appends two lists. Normally used via the ++
operator.
Appending lists takes time proportional to the length of the first list: O(|xs|)
.
Examples:
[1, 2, 3] ++ [4, 5] = [1, 2, 3, 4, 5]
.[] ++ [4, 5] = [4, 5]
.[1, 2, 3] ++ [] = [1, 2, 3]
.
Appends two lists. Normally used via the ++
operator.
Appending lists takes time proportional to the length of the first list: O(|xs|)
.
This is a tail-recursive version of List.append
.
Examples:
[1, 2, 3] ++ [4, 5] = [1, 2, 3, 4, 5]
.[] ++ [4, 5] = [4, 5]
.[1, 2, 3] ++ [] = [1, 2, 3]
.
Equations
- as.appendTR bs = as.reverse.reverseAux bs
Equations
- List.instAppend = { append := List.append }
flatten #
Concatenates a list of lists into a single list, preserving the order of the elements.
O(|flatten L|)
.
Examples:
Concatenates a list of lists into a single list, preserving the order of the elements.
O(|flatten L|)
.
Examples:
[["a"], ["b", "c"]].flatten = ["a", "b", "c"]
[["a"], [], ["b", "c"], ["d", "e", "f"]].flatten = ["a", "b", "c", "d", "e", "f"]
Equations
singleton #
Constructs a single-element list.
Examples:
List.singleton 5 = [5]
.List.singleton "green" = ["green"]
.List.singleton [1, 2, 3] = [[1, 2, 3]]
Equations
- List.singleton a = [a]
flatMap #
Applies a function that returns a list to each element of a list, and concatenates the resulting lists.
Examples:
[2, 3, 2].flatMap List.range = [0, 1, 0, 1, 2, 0, 1]
["red", "blue"].flatMap String.toList = ['r', 'e', 'd', 'b', 'l', 'u', 'e']
Equations
- List.flatMap b as = (List.map b as).flatten
Equations
Equations
replicate #
Creates a list that contains n
copies of a
.
List.replicate 5 "five" = ["five", "five", "five", "five", "five"]
List.replicate 0 "zero" = []
List.replicate 2 ' ' = [' ', ' ']
Equations
- List.replicate 0 x✝ = []
- List.replicate n.succ x✝ = x✝ :: List.replicate n x✝
Additional functions #
leftpad and rightpad #
Pads l : List α
on the left with repeated occurrences of a : α
until it is of length n
. If l
already has at least n
elements, it is returned unmodified.
Examples:
[1, 2, 3].leftpad 5 0 = [0, 0, 1, 2, 3]
["red", "green", "blue"].leftpad 4 "blank" = ["blank", "red", "green", "blue"]
["red", "green", "blue"].leftpad 3 "blank" = ["red", "green", "blue"]
["red", "green", "blue"].leftpad 1 "blank" = ["red", "green", "blue"]
Equations
- List.leftpad n a l = List.replicate (n - l.length) a ++ l
Pads l : List α
on the right with repeated occurrences of a : α
until it is of length n
. If
l
already has at least n
elements, it is returned unmodified.
Examples:
[1, 2, 3].rightpad 5 0 = [1, 2, 3, 0, 0]
["red", "green", "blue"].rightpad 4 "blank" = ["red", "green", "blue", "blank"]
["red", "green", "blue"].rightpad 3 "blank" = ["red", "green", "blue"]
["red", "green", "blue"].rightpad 1 "blank" = ["red", "green", "blue"]
Equations
- List.rightpad n a l = l ++ List.replicate (n - l.length) a
reduceOption #
Drop none
s from a list, and replace each remaining some a
with a
.
Equations
List membership #
EmptyCollection #
Equations
- List.instEmptyCollection = { emptyCollection := [] }
isEmpty #
elem #
Checks whether a
is an element of l
, using ==
to compare elements.
O(|l|)
. List.contains
is a synonym that takes the list before the element.
The preferred simp normal form is l.contains a
. When LawfulBEq α
is available,
l.contains a = true ↔ a ∈ l
and l.contains a = false ↔ a ∉ l
.
Example:
contains #
Checks whether a
is an element of as
, using ==
to compare elements.
O(|as|)
. List.elem
is a synonym that takes the element before the list.
The preferred simp normal form is l.contains a
, and when LawfulBEq α
is available,
l.contains a = true ↔ a ∈ l
and l.contains a = false ↔ a ∉ l
.
Examples:
[1, 4, 2, 3, 3, 7].contains 3 = true
List.contains [1, 4, 2, 3, 3, 7] 5 = false
Mem #
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 #
Extracts the first n
elements of xs
, or the whole list if n
is greater than xs.length
.
O(min n |xs|)
.
Examples:
drop #
Removes the first n
elements of the list xs
. Returns the empty list if n
is greater than the
length of the list.
O(min n |xs|)
.
Examples:
extract #
Returns the slice of l
from indices start
(inclusive) to stop
(exclusive).
Examples:
- [0, 1, 2, 3, 4, 5].extract 1 2 = [1]
- [0, 1, 2, 3, 4, 5].extract 2 2 = []
- [0, 1, 2, 3, 4, 5].extract 2 4 = [2, 3]
- [0, 1, 2, 3, 4, 5].extract 2 = [2, 3, 4, 5]
- [0, 1, 2, 3, 4, 5].extract (stop := 2) = [0, 1]
takeWhile #
Returns the longest initial segment of xs
for which p
returns true.
O(|xs|)
.
Examples:
[7, 6, 4, 8].takeWhile (· > 5) = [7, 6]
[7, 6, 6, 5].takeWhile (· > 5) = [7, 6, 6]
[7, 6, 6, 8].takeWhile (· > 5) = [7, 6, 6, 8]
Equations
- List.takeWhile p [] = []
- List.takeWhile p (a :: as) = match p a with | true => a :: List.takeWhile p as | false => []
dropWhile #
Removes the longest prefix of a list for which p
returns true
.
Elements are removed from the list until one is encountered for which p
returns false
. This
element and the remainder of the list are returned.
O(|l|)
.
Examples:
[1, 3, 2, 4, 2, 7, 4].dropWhile (· < 4) = [4, 2, 7, 4]
[8, 3, 2, 4, 2, 7, 4].dropWhile (· < 4) = [8, 3, 2, 4, 2, 7, 4]
[8, 3, 2, 4, 2, 7, 4].dropWhile (· < 100) = []
Equations
- List.dropWhile p [] = []
- List.dropWhile p (a :: as) = match p a with | true => List.dropWhile p as | false => a :: as
partition #
Returns a pair of lists that together contain all the elements of as
. The first list contains
those elements for which p
returns true
, and the second contains those for which p
returns
false
.
O(|l|)
. as.partition p
is equivalent to (as.filter p, as.filter (not ∘ p))
, but it is slightly
more efficient since it only has to do one pass over the list.
Examples:
[1, 2, 5, 2, 7, 7].partition (· > 2) = ([5, 7, 7], [1, 2, 2])
[1, 2, 5, 2, 7, 7].partition (fun _ => false) = ([], [1, 2, 5, 2, 7, 7])
[1, 2, 5, 2, 7, 7].partition (fun _ => true) = ([1, 2, 5, 2, 7, 7], [])
Equations
- List.partition p as = List.partition.loop p as ([], [])
dropLast #
Subset #
Equations
- List.instHasSubset = { Subset := List.Subset }
Equations
- x✝¹.instDecidableRelSubsetOfDecidableEq x✝ = List.decidableBAll (Membership.mem x✝) x✝¹
Sublist and isSublist #
The first list is a non-contiguous sub-list of the second list. Typically written with the <+
operator.
In other words, l₁ <+ l₂
means that l₁
can be transformed into l₂
by repeatedly inserting new
elements.
- 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₂
.
The first list is a non-contiguous sub-list of the second list. Typically written with the <+
operator.
In other words, l₁ <+ l₂
means that l₁
can be transformed into l₂
by repeatedly inserting new
elements.
Equations
- List.«term_<+_» = Lean.ParserDescr.trailingNode `List.«term_<+_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <+ ") (Lean.ParserDescr.cat `term 51))
True if the first list is a potentially non-contiguous sub-sequence of the second list, comparing
elements with the ==
operator.
The relation List.Sublist
is a logical characterization of this property.
Examples:
IsPrefix / isPrefixOf / isPrefixOf? #
The first list is a prefix of the second.
IsPrefix l₁ l₂
, written l₁ <+: l₂
, means that there exists some t : List α
such that l₂
has
the form l₁ ++ t
.
The function List.isPrefixOf
is a Boolean equivalent.
Conventions for notations in identifiers:
- The recommended spelling of
<+:
in identifiers isprefix
(notisPrefix
).
Instances For
The first list is a prefix of the second.
IsPrefix l₁ l₂
, written l₁ <+: l₂
, means that there exists some t : List α
such that l₂
has
the form l₁ ++ t
.
The function List.isPrefixOf
is a Boolean equivalent.
Conventions for notations in identifiers:
- The recommended spelling of
<+:
in identifiers isprefix
(notisPrefix
).
Equations
- List.«term_<+:_» = Lean.ParserDescr.trailingNode `List.«term_<+:_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <+: ") (Lean.ParserDescr.cat `term 51))
Checks whether the first list is a prefix of the second.
The relation List.IsPrefixOf
expresses this property with respect to logical equality.
Examples:
[1, 2].isPrefixOf [1, 2, 3] = true
[1, 2].isPrefixOf [1, 2] = true
[1, 2].isPrefixOf [1] = false
[1, 2].isPrefixOf [1, 1, 2, 3] = false
Equations
- [].isPrefixOf x✝ = true
- x✝.isPrefixOf [] = false
- (a :: as).isPrefixOf (b :: bs) = (a == b && as.isPrefixOf bs)
If the first list is a prefix of the second, returns the result of dropping the prefix.
In other words, isPrefixOf? l₁ l₂
returns some t
when l₂ == l₁ ++ t
.
Examples:
[1, 2].isPrefixOf? [1, 2, 3] = some [3]
[1, 2].isPrefixOf? [1, 2] = some []
[1, 2].isPrefixOf? [1] = none
[1, 2].isPrefixOf? [1, 1, 2, 3] = none
IsSuffix / isSuffixOf / isSuffixOf? #
Checks whether the first list is a suffix of the second.
The relation List.IsSuffixOf
expresses this property with respect to logical equality.
Examples:
[2, 3].isSuffixOf [1, 2, 3] = true
[2, 3].isSuffixOf [1, 2, 3, 4] = false
[2, 3].isSuffixOf [1, 2] = false
[2, 3].isSuffixOf [1, 1, 2, 3] = true
Equations
- l₁.isSuffixOf l₂ = l₁.reverse.isPrefixOf l₂.reverse
If the first list is a suffix of the second, returns the result of dropping the suffix from the second.
In other words, isSuffixOf? l₁ l₂
returns some t
when l₂ == t ++ l₁
.
Examples:
[2, 3].isSuffixOf? [1, 2, 3] = some [1]
[2, 3].isSuffixOf? [1, 2, 3, 4] = none
[2, 3].isSuffixOf? [1, 2] = none
[2, 3].isSuffixOf? [1, 1, 2, 3] = some [1, 1]
Equations
- l₁.isSuffixOf? l₂ = Option.map List.reverse (l₁.reverse.isPrefixOf? l₂.reverse)
The first list is a suffix of the second.
IsSuffix l₁ l₂
, written l₁ <:+ l₂
, means that there exists some t : List α
such that l₂
has
the form t ++ l₁
.
The function List.isSuffixOf
is a Boolean equivalent.
Conventions for notations in identifiers:
- The recommended spelling of
<:+
in identifiers issuffix
(notisSuffix
).
Instances For
The first list is a suffix of the second.
IsSuffix l₁ l₂
, written l₁ <:+ l₂
, means that there exists some t : List α
such that l₂
has
the form t ++ l₁
.
The function List.isSuffixOf
is a Boolean equivalent.
Conventions for notations in identifiers:
- The recommended spelling of
<:+
in identifiers issuffix
(notisSuffix
).
Equations
- List.«term_<:+_» = Lean.ParserDescr.trailingNode `List.«term_<:+_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <:+ ") (Lean.ParserDescr.cat `term 51))
IsInfix #
The first list is a contiguous sub-list of the second list. Typically written with the <:+:
operator.
In other words, l₁ <:+: l₂
means that there exist lists s : List α
and t : List α
such that
l₂
has the form s ++ l₁ ++ t
.
Conventions for notations in identifiers:
- The recommended spelling of
<:+:
in identifiers isinfix
(notisInfix
).
The first list is a contiguous sub-list of the second list. Typically written with the <:+:
operator.
In other words, l₁ <:+: l₂
means that there exist lists s : List α
and t : List α
such that
l₂
has the form s ++ l₁ ++ t
.
Conventions for notations in identifiers:
- The recommended spelling of
<:+:
in identifiers isinfix
(notisInfix
).
Equations
- List.«term_<:+:_» = Lean.ParserDescr.trailingNode `List.«term_<:+:_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " <:+: ") (Lean.ParserDescr.cat `term 51))
splitAt #
Splits a list at an index, resulting in the first n
elements of l
paired with the remaining
elements.
If n
is greater than the length of l
, then the resulting pair consists of l
and the empty
list. List.splitAt
is equivalent to a combination of List.take
and List.drop
, but it is more
efficient.
Examples:
["red", "green", "blue"].splitAt 2 = (["red", "green"], ["blue"])
["red", "green", "blue"].splitAt 3 = (["red", "green", "blue], [])
["red", "green", "blue"].splitAt 4 = (["red", "green", "blue], [])
Equations
- List.splitAt n l = List.splitAt.go l l n []
Auxiliary for splitAt
:
splitAt.go l xs n acc = (acc.reverse ++ take n xs, drop n xs)
if n < xs.length
,
and (l, [])
otherwise.
rotateLeft #
Rotates the elements of xs
to the left, moving i % xs.length
elements from the start of the list
to the end.
O(|xs|)
.
Examples:
[1, 2, 3, 4, 5].rotateLeft 3 = [4, 5, 1, 2, 3]
[1, 2, 3, 4, 5].rotateLeft 5 = [1, 2, 3, 4, 5]
[1, 2, 3, 4, 5].rotateLeft 1 = [2, 3, 4, 5, 1]
rotateRight #
Rotates the elements of xs
to the right, moving i % xs.length
elements from the end of the list
to the start.
After rotation, the element at xs[n]
is at index (i + n) % l.length
. O(|xs|)
.
Examples:
[1, 2, 3, 4, 5].rotateRight 3 = [3, 4, 5, 1, 2]
[1, 2, 3, 4, 5].rotateRight 5 = [1, 2, 3, 4, 5]
[1, 2, 3, 4, 5].rotateRight 1 = [5, 1, 2, 3, 4]
Pairwise, Nodup #
Each element of a list is related to all later elements of the list by R
.
Pairwise R l
means that all the elements of l
with earlier indexes are R
-related to all the
elements with later indexes.
For example, Pairwise (· ≠ ·) l
asserts that l
has no duplicates, and if Pairwise (· < ·) l
asserts that l
is (strictly) sorted.
Examples:
Pairwise (· < ·) [1, 2, 3] ↔ (1 < 2 ∧ 1 < 3) ∧ 2 < 3
Pairwise (· = ·) [1, 2, 3] = False
Pairwise (· ≠ ·) [1, 2, 3] = True
- nil
{α : Type u}
{R : α → α → Prop}
: 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') → Pairwise R l → Pairwise R (a :: l)
Instances For
Equations
- One or more equations did not get rendered due to their size.
- [].instDecidablePairwise = isTrue ⋯
The list has no duplicates: it contains every element at most once.
It is defined as Pairwise (· ≠ ·)
: each element is unequal to all other elements.
Equations
- List.Nodup = List.Pairwise fun (x1 x2 : α) => x1 ≠ x2
Instances For
Manipulating elements #
replace #
Replaces the first element of the list l
that is equal to a
with b
. If no element is equal to
a
, then the list is returned unchanged.
O(|l|)
.
Examples:
modify #
Replaces the n
th tail of l
with the result of applying f
to it. Returns the input without
using f
if the index is larger than the length of the List.
Examples:
["circle", "square", "triangle"].modifyTailIdx 1 List.reverse
["circle", "triangle", "square"]
["circle", "square", "triangle"].modifyTailIdx 1 (fun xs => xs ++ xs)
["circle", "square", "triangle", "square", "triangle"]
["circle", "square", "triangle"].modifyTailIdx 2 (fun xs => xs ++ xs)
["circle", "square", "triangle", "triangle"]
["circle", "square", "triangle"].modifyTailIdx 5 (fun xs => xs ++ xs)
["circle", "square", "triangle"]
Equations
- l.modifyTailIdx i f = List.modifyTailIdx.go f i l
Equations
- List.modifyTailIdx.go f 0 x✝ = f x✝
- List.modifyTailIdx.go f n.succ [] = []
- List.modifyTailIdx.go f n.succ (a :: as) = a :: List.modifyTailIdx.go f n as
Replace the head of the list with the result of applying f
to it. Returns the empty list if the
list is empty.
Examples:
[1, 2, 3].modifyHead (· * 10) = [10, 2, 3]
[].modifyHead (· * 10) = []
Equations
- List.modifyHead f [] = []
- List.modifyHead f (a :: as) = f a :: as
Replaces the element at the given index, if it exists, with the result of applying f
to it. If the
index is invalid, the list is returned unmodified.
Examples:
[1, 2, 3].modify 0 (· * 10) = [10, 2, 3]
[1, 2, 3].modify 2 (· * 10) = [1, 2, 30]
[1, 2, 3].modify 3 (· * 10) = [1, 2, 3]
Equations
- l.modify i f = l.modifyTailIdx i (List.modifyHead f)
insert #
Inserts an element into a list without duplication.
If the element is present in the list, the list is returned unmodified. Otherwise, the new element is inserted at the head of the list.
Examples:
Inserts an element into a list at the specified index. If the index is greater than the length of the list, then the list is returned unmodified.
In other words, the new element is inserted into the list l
after the first i
elements of l
.
Examples:
["tues", "thur", "sat"].insertIdx 1 "wed" = ["tues", "wed", "thur", "sat"]
["tues", "thur", "sat"].insertIdx 2 "wed" = ["tues", "thur", "wed", "sat"]
["tues", "thur", "sat"].insertIdx 3 "wed" = ["tues", "thur", "sat", "wed"]
["tues", "thur", "sat"].insertIdx 4 "wed" = ["tues", "thur", "sat"]
Equations
- xs.insertIdx i a = xs.modifyTailIdx i (List.cons a)
erase #
Removes the first occurrence of a
from l
. If a
does not occur in l
, the list is returned
unmodified.
O(|l|)
.
Examples:
Removes the first element of a list for which p
returns true
. If no element satisfies p
, then
the list is returned unchanged.
Examples:
[2, 1, 2, 1, 3, 4].eraseP (· < 2) = [2, 2, 1, 3, 4]
[2, 1, 2, 1, 3, 4].eraseP (· > 2) = [2, 1, 2, 1, 4]
[2, 1, 2, 1, 3, 4].eraseP (· > 8) = [2, 1, 2, 1, 3, 4]
Equations
- List.eraseP p [] = []
- List.eraseP p (a :: as) = bif p a then as else a :: List.eraseP p as
eraseIdx #
Removes the element at the specified index. If the index is out of bounds, the list is returned unmodified.
O(i)
.
Examples:
Finding elements
find? #
Returns the first element of the list for which the predicate p
returns true
, or none
if no
such element is found.
O(|l|)
.
Examples:
Equations
- List.find? p [] = none
- List.find? p (a :: as) = match p a with | true => some a | false => List.find? p as
findSome? #
Returns the first non-none
result of applying f
to each element of the list in order. Returns
none
if f
returns none
for all elements of the list.
O(|l|)
.
Examples:
[7, 6, 5, 8, 1, 2, 6].findSome? (fun x => if x < 5 then some (10 * x) else none) = some 10
[7, 6, 5, 8, 1, 2, 6].findSome? (fun x => if x < 1 then some (10 * x) else none) = none
Equations
- List.findSome? f [] = none
- List.findSome? f (a :: as) = match f a with | some b => some b | none => List.findSome? f as
findIdx #
Returns the index of the first element for which p
returns true
, or the length of the list if
there is no such element.
Examples:
Equations
- List.findIdx p l = List.findIdx.go p l 0
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)
idxOf #
Returns the index of the first element equal to a
, or the length of the list if no element is
equal to a
.
Examples:
["carrot", "potato", "broccoli"].idxOf "carrot" = 0
["carrot", "potato", "broccoli"].idxOf "broccoli" = 2
["carrot", "potato", "broccoli"].idxOf "tomato" = 3
["carrot", "potato", "broccoli"].idxOf "anything else" = 3
Equations
- List.idxOf a = List.findIdx fun (x : α) => x == a
Returns the index of the first element equal to a
, or the length of the list otherwise.
Equations
findIdx? #
Returns the index of the first element for which p
returns true
, or none
if there is no such
element.
Examples:
Equations
- List.findIdx? p l = List.findIdx?.go p l 0
idxOf? #
Returns the index of the first element equal to a
, or none
if no element is equal to a
.
Examples:
["carrot", "potato", "broccoli"].idxOf? "carrot" = some 0
["carrot", "potato", "broccoli"].idxOf? "broccoli" = some 2
["carrot", "potato", "broccoli"].idxOf? "tomato" = none
["carrot", "potato", "broccoli"].idxOf? "anything else" = none
Equations
- List.idxOf? a = List.findIdx? fun (x : α) => x == a
Return the index of the first occurrence of a
in the list.
Equations
findFinIdx? #
Returns the index of the first element for which p
returns true
, or none
if there is no such
element. The index is returned as a Fin
, which guarantees that it is in bounds.
Examples:
[7, 6, 5, 8, 1, 2, 6].findFinIdx? (· < 5) = some (4 : Fin 7)
[7, 6, 5, 8, 1, 2, 6].findFinIdx? (· < 1) = none
Equations
- List.findFinIdx? p l = List.findFinIdx?.go p l l 0 ⋯
finIdxOf? #
Returns the index of the first element equal to a
, or the length of the list if no element is
equal to a
. The index is returned as a Fin
, which guarantees that it is in bounds.
Examples:
["carrot", "potato", "broccoli"].finIdxOf? "carrot" = some 0
["carrot", "potato", "broccoli"].finIdxOf? "broccoli" = some 2
["carrot", "potato", "broccoli"].finIdxOf? "tomato" = none
["carrot", "potato", "broccoli"].finIdxOf? "anything else" = none
Equations
- List.finIdxOf? a = List.findFinIdx? fun (x : α) => x == a
countP #
Counts the number of elements in the list l
that satisfy the Boolean predicate p
.
Examples:
[1, 2, 3, 4, 5].countP (· % 2 == 0) = 2
[1, 2, 3, 4, 5].countP (· < 5) = 4
[1, 2, 3, 4, 5].countP (· > 5) = 0
Equations
- List.countP p l = List.countP.go p l 0
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✝
count #
Counts the number of times an element occurs in a list.
Examples:
Equations
- List.count a = List.countP fun (x : α) => x == a
lookup #
Treats the list as an association list that maps keys to values, returning the first value whose key is equal to the specified key.
O(|l|)
.
Examples:
Permutations #
Perm #
Two lists are permutations of each other if they contain the same elements, each occurring the same number of times but not necessarily in the same order.
One list can be proven to be a permutation of another by showing how to transform one into the other by repeatedly swapping adjacent elements.
List.isPerm
is a Boolean equivalent of this relation.
- nil
{α : Type u}
: [].Perm []
The empty list is a permutation of the empty list:
[] ~ []
. - cons
{α : Type u}
(x : α)
{l₁ l₂ : List α}
: l₁.Perm l₂ → (x :: l₁).Perm (x :: l₂)
If one list is a permutation of the other, adding the same element as the head of each yields lists that are permutations of each other:
l₁ ~ l₂ → x::l₁ ~ x::l₂
. - swap
{α : Type u}
(x y : α)
(l : List α)
: (y :: x :: l).Perm (x :: y :: l)
If two lists are identical except for having their first two elements swapped, then they are permutations of each other:
x::y::l ~ y::x::l
. - trans
{α : Type u}
{l₁ l₂ l₃ : List α}
: l₁.Perm l₂ → l₂.Perm l₃ → l₁.Perm l₃
Permutation is transitive:
l₁ ~ l₂ → l₂ ~ l₃ → l₁ ~ l₃
.
Instances For
Two lists are permutations of each other if they contain the same elements, each occurring the same number of times but not necessarily in the same order.
One list can be proven to be a permutation of another by showing how to transform one into the other by repeatedly swapping adjacent elements.
List.isPerm
is a Boolean equivalent of this relation.
Equations
- List.«term_~_» = Lean.ParserDescr.trailingNode `List.«term_~_» 50 50 (Lean.ParserDescr.binary `andthen (Lean.ParserDescr.symbol " ~ ") (Lean.ParserDescr.cat `term 51))
isPerm #
Returns true
if l₁
and l₂
are permutations of each other. O(|l₁| * |l₂|)
.
The relation List.Perm
is a logical characterization of permutations. When the BEq α
instance
corresponds to DecidableEq α
, isPerm l₁ l₂ ↔ l₁ ~ l₂
(use the theorem isPerm_iff
).
Logical operations #
any #
Returns true
if p
returns true
for any element of l
.
O(|l|)
. Short-circuits upon encountering the first true
.
Examples:
all #
Returns true
if p
returns true
for every element of l
.
O(|l|)
. Short-circuits upon encountering the first false
.
Examples:
or #
Returns true
if true
is an element of the list bs
.
O(|bs|)
. Short-circuits at the first true
value.
and #
Zippers #
zipWith #
Applies a function to the corresponding elements of two lists, stopping at the end of the shorter list.
O(min |xs| |ys|)
.
Examples:
[1, 2].zipWith (· + ·) [5, 6] = [6, 8]
[1, 2, 3].zipWith (· + ·) [5, 6, 10] = [6, 8, 13]
[].zipWith (· + ·) [5, 6] = []
[x₁, x₂, x₃].zipWith f [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✝ = []
zip #
Combines two lists into a list of pairs in which the first and second components are the corresponding elements of each list. The resulting list is the length of the shorter of the input lists.
O(min |xs| |ys|)
.
Examples:
["Mon", "Tue", "Wed"].zip [1, 2, 3] = [("Mon", 1), ("Tue", 2), ("Wed", 3)]
["Mon", "Tue", "Wed"].zip [1, 2] = [("Mon", 1), ("Tue", 2)]
[x₁, x₂, x₃].zip [y₁, y₂, y₃, y₄] = [(x₁, y₁), (x₂, y₂), (x₃, y₃)]
Equations
zipWithAll #
Applies a function to the corresponding elements of both lists, stopping when there are no more
elements in either list. If one list is shorter than the other, the function is passed none
for
the missing elements.
Examples:
[1, 6].zipWithAll min [5, 2] = [some 1, some 2]
[1, 2, 3].zipWithAll Prod.mk [5, 6] = [(some 1, some 5), (some 2, some 6), (some 3, none)]
[x₁, x₂].zipWithAll f [y] = [f (some x₁) (some y), f (some x₂) none]
unzip #
Separates a list of pairs into two lists that contain the respective first and second components.
O(|l|)
.
Examples:
Ranges and enumeration #
range #
Returns a list of the numbers from 0
to n
exclusive, in increasing order.
O(n)
.
Examples:
Equations
- List.range n = List.range.loop n []
Equations
- List.range.loop 0 x✝ = x✝
- List.range.loop n.succ x✝ = List.range.loop n (n :: x✝)
range' #
Returns a list of the numbers with the given length len
, starting at start
and increasing by
step
at each element.
In other words, List.range' start len step
is [start, start+step, ..., start+(len-1)*step]
.
Examples:
List.range' 0 3 (step := 1) = [0, 1, 2]
List.range' 0 3 (step := 2) = [0, 2, 4]
List.range' 0 4 (step := 2) = [0, 2, 4, 6]
List.range' 3 4 (step := 2) = [3, 5, 7, 9]
Equations
- List.range' x✝¹ 0 x✝ = []
- List.range' x✝¹ n.succ x✝ = x✝¹ :: List.range' (x✝¹ + x✝) n x✝
iota #
zipIdx #
Pairs each element of a list with its index, optionally starting from an index other than 0
.
O(|l|)
.
Examples:
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
enum #
Minima and maxima #
min? #
Returns the smallest element of the list if it is not empty, or none
if it is empty.
Examples:
Equations
max? #
Returns the largest element of the list if it is not empty, or none
if it is empty.
Examples:
Equations
Other list operations #
The functions are currently mostly used in meta code, and do not have sufficient API developed for verification work.
intersperse #
Alternates the elements of l
with sep
.
O(|l|)
.
List.intercalate
is a similar function that alternates a separator list with elements of a list of
lists.
Examples:
List.intersperse "then" [] = []
List.intersperse "then" ["walk"] = ["walk"]
List.intersperse "then" ["walk", "run"] = ["walk", "then", "run"]
List.intersperse "then" ["walk", "run", "rest"] = ["walk", "then", "run", "then", "rest"]
Equations
- List.intersperse sep [] = []
- List.intersperse sep [head] = [head]
- List.intersperse sep (a :: as) = a :: sep :: List.intersperse sep as
intercalate #
Alternates the lists in xs
with the separator sep
, appending them. The resulting list is
flattened.
O(|xs|)
.
List.intersperse
is a similar function that alternates a separator element with the elements of a
list.
Examples:
List.intercalate sep [] = []
List.intercalate sep [a] = a
List.intercalate sep [a, b] = a ++ sep ++ b
List.intercalate sep [a, b, c] = a ++ sep ++ b ++ sep ++ c
Equations
- sep.intercalate xs = (List.intersperse sep xs).flatten
eraseDups #
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✝)
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✝)
span #
Splits a list into the the longest initial segment for which p
returns true
, paired with the
remainder of the list.
O(|l|)
.
Examples:
[6, 8, 9, 5, 2, 9].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].span (· > 0) = ([6, 8, 9, 5, 2, 9], [])
Equations
- List.span p as = List.span.loop p as []
splitBy #
Splits a list into the longest segments in which each pair of adjacent elements are related by R
.
O(|l|)
.
Examples:
[1, 1, 2, 2, 2, 3, 2].splitBy (· == ·) = [[1, 1], [2, 2, 2], [3], [2]]
[1, 2, 5, 4, 5, 1, 4].splitBy (· < ·) = [[1, 2, 5], [4, 5], [1, 4]]
[1, 2, 5, 4, 5, 1, 4].splitBy (fun _ _ => true) = [[1, 2, 5, 4, 5, 1, 4]]
[1, 2, 5, 4, 5, 1, 4].splitBy (fun _ _ => false) = [[1], [2], [5], [4], [5], [1], [4]]
Equations
- List.splitBy R [] = []
- List.splitBy R (a :: as) = List.splitBy.loop R as a [] []
The arguments of splitBy.loop l b g gs
represent the following:
Splits a list into the longest segments in which each pair of adjacent elements are related by R
.
O(|l|)
.
Examples:
[1, 1, 2, 2, 2, 3, 2].splitBy (· == ·) = [[1, 1], [2, 2, 2], [3], [2]]
[1, 2, 5, 4, 5, 1, 4].splitBy (· < ·) = [[1, 2, 5], [4, 5], [1, 4]]
[1, 2, 5, 4, 5, 1, 4].splitBy (fun _ _ => true) = [[1, 2, 5, 4, 5, 1, 4]]
[1, 2, 5, 4, 5, 1, 4].splitBy (fun _ _ => false) = [[1], [2], [5], [4], [5], [1], [4]]
Equations
removeAll #
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 #
Applies a function to each element of the list, returning the resulting list of values.
O(|l|)
. This is the tail-recursive variant of List.map
, used in runtime code.
Examples:
[a, b, c].mapTR f = [f a, f b, f c]
[].mapTR Nat.succ = []
["one", "two", "three"].mapTR (·.length) = [3, 3, 5]
["one", "two", "three"].mapTR (·.reverse) = ["eno", "owt", "eerht"]
Equations
- List.mapTR f as = List.mapTR.loop f as []
Equations
- List.mapTR.loop f [] x✝ = x✝.reverse
- List.mapTR.loop f (a :: as) x✝ = List.mapTR.loop f as (f a :: x✝)
filter #
Returns the list of elements in l
for which p
returns true
.
O(|l|)
. This is a tail-recursive version of List.filter
, used at runtime.
Examples:
[1, 2, 5, 2, 7, 7].filterTR (· > 2) = [5, 7, 7]
[1, 2, 5, 2, 7, 7].filterTR (fun _ => false) = []
[1, 2, 5, 2, 7, 7].filterTR (fun _ => true) = * [1, 2, 5, 2, 7, 7]
Equations
- List.filterTR p as = List.filterTR.loop p as []
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✝
replicate #
Creates a list that contains n
copies of a
.
This is a tail-recursive version of List.replicate
.
List.replicateTR 5 "five" = ["five", "five", "five", "five", "five"]
List.replicateTR 0 "zero" = []
List.replicateTR 2 ' ' = [' ', ' ']
Equations
- List.replicateTR n a = List.replicateTR.loop a n []
Equations
- List.replicateTR.loop a 0 x✝ = x✝
- List.replicateTR.loop a n.succ x✝ = List.replicateTR.loop a n (a :: x✝)
Additional functions #
leftpad #
Pads l : List α
on the left with repeated occurrences of a : α
until it is of length n
. If l
already has at least n
elements, it is returned unmodified.
This is a tail-recursive version of List.leftpad
, used at runtime.
Examples:
[1, 2, 3].leftPadTR 5 0 = [0, 0, 1, 2, 3]
["red", "green", "blue"].leftPadTR 4 "blank" = ["blank", "red", "green", "blue"]
["red", "green", "blue"].leftPadTR 3 "blank" = ["red", "green", "blue"]
["red", "green", "blue"].leftPadTR 1 "blank" = ["red", "green", "blue"]
Equations
- List.leftpadTR n a l = List.replicateTR.loop a (n - l.length) l
Zippers #
unzip #
Separates a list of pairs into two lists that contain the respective first and second components.
O(|l|)
. This is a tail-recursive version of List.unzip
that's used at runtime.
Examples:
Ranges and enumeration #
range' #
Returns a list of the numbers with the given length len
, starting at start
and increasing by
step
at each element.
In other words, List.range'TR start len step
is [start, start+step, ..., start+(len-1)*step]
.
This is a tail-recursive version of List.range'
.
Examples:
List.range'TR 0 3 (step := 1) = [0, 1, 2]
List.range'TR 0 3 (step := 2) = [0, 2, 4]
List.range'TR 0 4 (step := 2) = [0, 2, 4, 6]
List.range'TR 3 4 (step := 2) = [3, 5, 7, 9]
Equations
- List.range'TR s n step = List.range'TR.go step n (s + step * n) []
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✝)
iota #
Tail-recursive version of List.iota
.
Equations
- List.iotaTR n = List.iotaTR.go n []
Equations
- List.iotaTR.go 0 x✝ = x✝.reverse
- List.iotaTR.go n.succ x✝ = List.iotaTR.go n (n.succ :: x✝)
Other list operations #
intersperse #
Alternates the elements of l
with sep
.
O(|l|)
.
This is a tail-recursive version of List.intersperse
, used at runtime.
Examples:
List.intersperseTR "then" [] = []
List.intersperseTR "then" ["walk"] = ["walk"]
List.intersperseTR "then" ["walk", "run"] = ["walk", "then", "run"]
List.intersperseTR "then" ["walk", "run", "rest"] = ["walk", "then", "run", "then", "rest"]