/- Copyright (c) 2016 Microsoft Corporation. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Author: Leonardo de Moura -/ prelude import Init.SimpLemmas import Init.Data.Nat.Basic import Init.Data.List.Notation /-! # 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 to `Init.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`, `fold`, and `concat` have already been defined in `Init.Prelude`. The operations are organized as follow: * Equality: `beq`, `isEqv`. * Lexicographic ordering: `lt`, `le`, and instances. * Basic operations: `map`, `filter`, `filterMap`, `foldr`, `append`, `join`, `pure`, `bind`, `replicate`, and `reverse`. * List membership: `isEmpty`, `elem`, `contains`, `mem` (and the `∈` notation), and decidability for predicates quantifying over membership in a `List`. * Sublists: `take`, `drop`, `takeWhile`, `dropWhile`, `partition`, `dropLast`, `isPrefixOf`, `isPrefixOf?`, `isSuffixOf`, `isSuffixOf?`, `Subset`, `Sublist`, `rotateLeft` and `rotateRight`. * Manipulating elements: `replace`, `insert`, `erase`, `eraseIdx`, `find?`, `findSome?`, and `lookup`. * Logic: `any`, `all`, `or`, and `and`. * Zippers: `zipWith`, `zip`, `zipWithAll`, and `unzip`. * Ranges and enumeration: `range`, `iota`, `enumFrom`, and `enum`. * Minima and maxima: `minimum?` and `maximum?`. * Other functions: `intersperse`, `intercalate`, `eraseDups`, `eraseReps`, `span`, `groupBy`, `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?`, and `getLastD`. * Head and tail: `head`, `head!`, `head?`, `headD`, `tail!`, `tail?`, and `tailD`. * Other operations on sublists: `partitionMap`, `rotateLeft`, and `rotateRight`. -/ set_option linter.missingDocs true -- keep it documented open Decidable List universe u v w variable {α : Type u} {β : Type v} {γ : Type w} namespace List /-! ## Preliminaries from `Init.Prelude` -/ /-! ### length -/ @[simp] theorem length_nil : length ([] : List α) = 0 := rfl @[simp 1100] theorem length_singleton (a : α) : length [a] = 1 := rfl @[simp] theorem length_cons {α} (a : α) (as : List α) : (cons a as).length = as.length + 1 := rfl /-! ### set -/ @[simp] theorem length_set (as : List α) (i : Nat) (a : α) : (as.set i a).length = as.length := by induction as generalizing i with | nil => rfl | cons x xs ih => cases i with | zero => rfl | succ i => simp [set, ih] /-! ### foldl -/ -- As `List.foldl` is defined in `Init.Prelude`, we write the basic simplification lemmas here. @[simp] theorem foldl_nil : [].foldl f b = b := rfl @[simp] theorem foldl_cons (l : List α) (b : β) : (a :: l).foldl f b = l.foldl f (f b a) := rfl /-! ### concat -/ @[simp high] theorem length_concat (as : List α) (a : α) : (concat as a).length = as.length + 1 := by induction as with | nil => rfl | cons _ xs ih => simp [concat, ih] theorem of_concat_eq_concat {as bs : List α} {a b : α} (h : as.concat a = bs.concat b) : as = bs ∧ a = b := by match as, bs with | [], [] => simp [concat] at h; simp [h] | [_], [] => simp [concat] at h | _::_::_, [] => simp [concat] at h | [], [_] => simp [concat] at h | [], _::_::_ => simp [concat] at h | _::_, _::_ => simp [concat] at h; simp [h]; apply of_concat_eq_concat h.2 /-! ## Equality -/ /-- The equality relation on lists asserts that they have the same length and they are pairwise `BEq`. -/ protected def beq [BEq α] : List α → List α → Bool | [], [] => true | a::as, b::bs => a == b && List.beq as bs | _, _ => false instance [BEq α] : BEq (List α) := ⟨List.beq⟩ instance [BEq α] [LawfulBEq α] : LawfulBEq (List α) where eq_of_beq {as bs} := by induction as generalizing bs with | nil => intro h; cases bs <;> first | rfl | contradiction | cons a as ih => cases bs with | nil => intro h; contradiction | cons b bs => simp [show (a::as == b::bs) = (a == b && as == bs) from rfl, -and_imp] intro ⟨h₁, h₂⟩ exact ⟨h₁, ih h₂⟩ rfl {as} := by induction as with | nil => rfl | cons a as ih => simp [BEq.beq, List.beq, LawfulBEq.rfl]; exact ih /-- `O(min |as| |bs|)`. Returns true if `as` and `bs` have the same length, and they are pairwise related by `eqv`. -/ @[specialize] def isEqv : (as bs : List α) → (eqv : α → α → Bool) → Bool | [], [], _ => true | a::as, b::bs, eqv => eqv a b && isEqv as bs eqv | _, _, _ => false @[simp] theorem isEqv_nil_nil : isEqv ([] : List α) [] eqv = true := rfl @[simp] theorem isEqv_nil_cons : isEqv ([] : List α) (a::as) eqv = false := rfl @[simp] theorem isEqv_cons_nil : isEqv (a::as : List α) [] eqv = false := rfl theorem isEqv_cons₂ : isEqv (a::as) (b::bs) eqv = (eqv a b && isEqv as bs eqv) := rfl /-! ## Lexicographic ordering -/ /-- The lexicographic order on lists. `[] < a::as`, and `a::as < b::bs` if `a < b` or if `a` and `b` are equivalent and `as < bs`. -/ inductive lt [LT α] : List α → List α → Prop where /-- `[]` is the smallest element in the order. -/ | nil (b : α) (bs : List α) : lt [] (b::bs) /-- If `a < b` then `a::as < b::bs`. -/ | head {a : α} (as : List α) {b : α} (bs : List α) : a < b → lt (a::as) (b::bs) /-- If `a` and `b` are equivalent and `as < bs`, then `a::as < b::bs`. -/ | tail {a : α} {as : List α} {b : α} {bs : List α} : ¬ a < b → ¬ b < a → lt as bs → lt (a::as) (b::bs) instance [LT α] : LT (List α) := ⟨List.lt⟩ instance hasDecidableLt [LT α] [h : DecidableRel (α := α) (· < ·)] : (l₁ l₂ : List α) → Decidable (l₁ < l₂) | [], [] => isFalse nofun | [], _::_ => isTrue (List.lt.nil _ _) | _::_, [] => isFalse nofun | a::as, b::bs => match h a b with | isTrue h₁ => isTrue (List.lt.head _ _ h₁) | isFalse h₁ => match h b a with | isTrue h₂ => isFalse (fun h => match h with | List.lt.head _ _ h₁' => absurd h₁' h₁ | List.lt.tail _ h₂' _ => absurd h₂ h₂') | isFalse h₂ => match hasDecidableLt as bs with | isTrue h₃ => isTrue (List.lt.tail h₁ h₂ h₃) | isFalse h₃ => isFalse (fun h => match h with | List.lt.head _ _ h₁' => absurd h₁' h₁ | List.lt.tail _ _ h₃' => absurd h₃' h₃) /-- The lexicographic order on lists. -/ @[reducible] protected def le [LT α] (a b : List α) : Prop := ¬ b < a instance [LT α] : LE (List α) := ⟨List.le⟩ instance [LT α] [DecidableRel ((· < ·) : α → α → Prop)] : (l₁ l₂ : List α) → Decidable (l₁ ≤ l₂) := fun _ _ => inferInstanceAs (Decidable (Not _)) /-! ## 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!`. -/ def get? : (as : List α) → (i : Nat) → Option α | a::_, 0 => some a | _::as, n+1 => get? as n | _, _ => none @[simp] theorem get?_nil : @get? α [] n = none := rfl @[simp] theorem get?_cons_zero : @get? α (a::l) 0 = some a := rfl @[simp] theorem get?_cons_succ : @get? α (a::l) (n+1) = get? l n := rfl theorem ext_get? : ∀ {l₁ l₂ : List α}, (∀ n, l₁.get? n = l₂.get? n) → l₁ = l₂ | [], [], _ => rfl | a :: l₁, [], h => nomatch h 0 | [], a' :: l₂, h => nomatch h 0 | a :: l₁, a' :: l₂, h => by have h0 : some a = some a' := h 0 injection h0 with aa; simp only [aa, ext_get? fun n => h (n+1)] /-- Deprecated alias for `ext_get?`. The preferred extensionality theorem is now `ext_getElem?`. -/ @[deprecated (since := "2024-06-07")] abbrev ext := @ext_get? /-! ### getD -/ /-- Returns the `i`-th element in the list (zero-based). If the index is out of bounds (`i ≥ as.length`), this function returns `fallback`. See also `get?` and `get!`. -/ def getD (as : List α) (i : Nat) (fallback : α) : α := (as.get? i).getD fallback @[simp] theorem getD_nil : getD [] n d = d := rfl @[simp] theorem getD_cons_zero : getD (x :: xs) 0 d = x := rfl @[simp] theorem getD_cons_succ : getD (x :: xs) (n + 1) d = getD xs n d := rfl /-! ### getLast -/ /-- Returns the last element of a non-empty list. -/ def getLast : ∀ (as : List α), as ≠ [] → α | [], h => absurd rfl h | [a], _ => a | _::b::as, _ => getLast (b::as) (fun h => List.noConfusion h) /-! ### getLast? -/ /-- Returns the last element in the list. If the list is empty, this function returns `none`. Also see `getLastD` and `getLast!`. -/ def getLast? : List α → Option α | [] => none | a::as => some (getLast (a::as) (fun h => List.noConfusion h)) @[simp] theorem getLast?_nil : @getLast? α [] = none := rfl /-! ### getLastD -/ /-- Returns the last element in the list. If the list is empty, this function returns `fallback`. Also see `getLast?` and `getLast!`. -/ def getLastD : (as : List α) → (fallback : α) → α | [], a₀ => a₀ | a::as, _ => getLast (a::as) (fun h => List.noConfusion h) @[simp] theorem getLastD_nil (a) : @getLastD α [] a = a := rfl @[simp] theorem getLastD_cons (a b l) : @getLastD α (b::l) a = getLastD l b := by cases l <;> rfl /-! ## Head and tail -/ /-! ### head -/ /-- Returns the first element of a non-empty list. -/ def head : (as : List α) → as ≠ [] → α | a::_, _ => a @[simp] theorem head_cons : @head α (a::l) h = a := rfl /-! ### head? -/ /-- Returns the first element in the list. If the list is empty, this function returns `none`. Also see `headD` and `head!`. -/ def head? : List α → Option α | [] => none | a::_ => some a @[simp] theorem head?_nil : @head? α [] = none := rfl @[simp] theorem head?_cons : @head? α (a::l) = some a := rfl /-! ### headD -/ /-- Returns the first element in the list. If the list is empty, this function returns `fallback`. Also see `head?` and `head!`. -/ def headD : (as : List α) → (fallback : α) → α | [], fallback => fallback | a::_, _ => a @[simp 1100] theorem headD_nil : @headD α [] d = d := rfl @[simp 1100] theorem headD_cons : @headD α (a::l) d = a := rfl /-! ### tail? -/ /-- Drops the first element of the list. If the list is empty, this function returns `none`. Also see `tailD` and `tail!`. -/ def tail? : List α → Option (List α) | [] => none | _::as => some as @[simp] theorem tail?_nil : @tail? α [] = none := rfl @[simp] theorem tail?_cons : @tail? α (a::l) = some l := rfl /-! ### tailD -/ /-- Drops the first element of the list. If the list is empty, this function returns `fallback`. Also see `head?` and `head!`. -/ def tailD (list fallback : List α) : List α := match list with | [] => fallback | _ :: tl => tl @[simp 1100] theorem tailD_nil : @tailD α [] l' = l' := rfl @[simp 1100] theorem tailD_cons : @tailD α (a::l) l' = l := rfl /-! ## Basic `List` operations. We define the basic functional programming operations on `List`: `map`, `filter`, `filterMap`, `foldr`, `append`, `join`, `pure`, `bind`, `replicate`, and `reverse`. -/ /-! ### map -/ /-- `O(|l|)`. `map f l` applies `f` to each element of the list. * `map f [a, b, c] = [f a, f b, f c]` -/ @[specialize] def map (f : α → β) : List α → List β | [] => [] | a::as => f a :: map f as @[simp] theorem map_nil {f : α → β} : map f [] = [] := rfl @[simp] theorem map_cons (f : α → β) a l : map f (a :: l) = f a :: map f l := rfl /-! ### 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] ``` -/ def filter (p : α → Bool) : List α → List α | [] => [] | a::as => match p a with | true => a :: filter p as | false => filter p as @[simp] theorem filter_nil (p : α → Bool) : filter p [] = [] := rfl /-! ### 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] ``` -/ @[specialize] def filterMap (f : α → Option β) : List α → List β | [] => [] | a::as => match f a with | none => filterMap f as | some b => b :: filterMap f as @[simp] theorem filterMap_nil (f : α → Option β) : filterMap f [] = [] := rfl theorem filterMap_cons (f : α → Option β) (a : α) (l : List α) : filterMap f (a :: l) = match f a with | none => filterMap f l | some b => b :: filterMap f l := rfl /-! ### 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` -/ @[specialize] def foldr (f : α → β → β) (init : β) : List α → β | [] => init | a :: l => f a (foldr f init l) @[simp] theorem foldr_nil : [].foldr f b = b := rfl @[simp] theorem foldr_cons (l : List α) : (a :: l).foldr f b = f a (l.foldr f b) := rfl /-! ### reverse -/ /-- Auxiliary for `List.reverse`. `List.reverseAux l r = l.reverse ++ r`, but it is defined directly. -/ def reverseAux : List α → List α → List α | [], r => r | a::l, r => reverseAux l (a::r) @[simp] theorem reverseAux_nil : reverseAux [] r = r := rfl @[simp] theorem reverseAux_cons : reverseAux (a::l) r = reverseAux l (a::r) := rfl /-- `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. -/ def reverse (as : List α) : List α := reverseAux as [] @[simp] theorem reverse_nil : reverse ([] : List α) = [] := rfl theorem reverseAux_reverseAux (as bs cs : List α) : reverseAux (reverseAux as bs) cs = reverseAux bs (reverseAux (reverseAux as []) cs) := by induction as generalizing bs cs with | nil => rfl | cons a as ih => simp [reverseAux, ih (a::bs), ih [a]] /-! ### append -/ /-- `O(|xs|)`: append two lists. `[1, 2, 3] ++ [4, 5] = [1, 2, 3, 4, 5]`. It takes time proportional to the first list. -/ protected def append : (xs ys : List α) → List α | [], bs => bs | a::as, bs => a :: List.append as bs /-- 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. -/ def appendTR (as bs : List α) : List α := reverseAux as.reverse bs @[csimp] theorem append_eq_appendTR : @List.append = @appendTR := by apply funext; intro α; apply funext; intro as; apply funext; intro bs simp [appendTR, reverse] induction as with | nil => rfl | cons a as ih => rw [reverseAux, reverseAux_reverseAux] simp [List.append, ih, reverseAux] instance : Append (List α) := ⟨List.append⟩ @[simp] theorem append_eq (as bs : List α) : List.append as bs = as ++ bs := rfl @[simp] theorem nil_append (as : List α) : [] ++ as = as := rfl @[simp] theorem cons_append (a : α) (as bs : List α) : (a::as) ++ bs = a::(as ++ bs) := rfl @[simp] theorem append_nil (as : List α) : as ++ [] = as := by induction as with | nil => rfl | cons a as ih => simp_all only [HAppend.hAppend, Append.append, List.append] instance : Std.LawfulIdentity (α := List α) (· ++ ·) [] where left_id := nil_append right_id := append_nil @[simp] theorem length_append (as bs : List α) : (as ++ bs).length = as.length + bs.length := by induction as with | nil => simp | cons _ as ih => simp [ih, Nat.succ_add] @[simp] theorem append_assoc (as bs cs : List α) : (as ++ bs) ++ cs = as ++ (bs ++ cs) := by induction as with | nil => rfl | cons a as ih => simp [ih] instance : Std.Associative (α := List α) (· ++ ·) := ⟨append_assoc⟩ theorem append_cons (as : List α) (b : α) (bs : List α) : as ++ b :: bs = as ++ [b] ++ bs := by simp @[simp] theorem concat_eq_append (as : List α) (a : α) : as.concat a = as ++ [a] := by induction as <;> simp [concat, *] theorem reverseAux_eq_append (as bs : List α) : reverseAux as bs = reverseAux as [] ++ bs := by induction as generalizing bs with | nil => simp [reverseAux] | cons a as ih => simp [reverseAux] rw [ih (a :: bs), ih [a], append_assoc] rfl @[simp] theorem reverse_cons (a : α) (as : List α) : reverse (a :: as) = reverse as ++ [a] := by simp [reverse, reverseAux] rw [← reverseAux_eq_append] /-! ### join -/ /-- `O(|join L|)`. `join L` concatenates all the lists in `L` into one list. * `join [[a], [], [b, c], [d, e, f]] = [a, b, c, d, e, f]` -/ def join : List (List α) → List α | [] => [] | a :: as => a ++ join as @[simp] theorem join_nil : List.join ([] : List (List α)) = [] := rfl @[simp] theorem join_cons : (l :: ls).join = l ++ ls.join := rfl /-! ### pure -/ /-- `pure x = [x]` is the `pure` operation of the list monad. -/ @[inline] protected def pure {α : Type u} (a : α) : List α := [a] /-! ### bind -/ /-- `bind xs f` is the bind operation of the list monad. It applies `f` to each element of `xs` to get a list of lists, and then concatenates them all together. * `[2, 3, 2].bind range = [0, 1, 0, 1, 2, 0, 1]` -/ @[inline] protected def bind {α : Type u} {β : Type v} (a : List α) (b : α → List β) : List β := join (map b a) @[simp] theorem bind_nil (f : α → List β) : List.bind [] f = [] := by simp [join, List.bind] @[simp] theorem bind_cons x xs (f : α → List β) : List.bind (x :: xs) f = f x ++ List.bind xs f := by simp [join, List.bind] set_option linter.missingDocs false in @[deprecated bind_nil (since := "2024-06-15")] abbrev nil_bind := @bind_nil set_option linter.missingDocs false in @[deprecated bind_cons (since := "2024-06-15")] abbrev cons_bind := @bind_cons /-! ### replicate -/ /-- `replicate n a` is `n` copies of `a`: * `replicate 5 a = [a, a, a, a, a]` -/ def replicate : (n : Nat) → (a : α) → List α | 0, _ => [] | n+1, a => a :: replicate n a @[simp] theorem replicate_zero : replicate 0 a = [] := rfl theorem replicate_succ (a : α) (n) : replicate (n+1) a = a :: replicate n a := rfl @[simp] theorem length_replicate (n : Nat) (a : α) : (replicate n a).length = n := by induction n with | zero => simp | succ n ih => simp only [ih, replicate_succ, length_cons, Nat.succ_eq_add_one] /-! ## List membership * `L.contains a : Bool` determines, using a `[BEq α]` instance, whether `L` contains an element `· == a`. * `a ∈ L : Prop` is the proposition (only decidable if `α` has decidable equality) that `L` contains an element `· = a`. -/ /-! ### EmptyCollection -/ instance : EmptyCollection (List α) := ⟨List.nil⟩ @[simp] theorem empty_eq : (∅ : List α) = [] := rfl /-! ### isEmpty -/ /-- `O(1)`. `isEmpty l` is true if the list is empty. * `isEmpty [] = true` * `isEmpty [a] = false` * `isEmpty [a, b] = false` -/ def isEmpty : List α → Bool | [] => true | _ :: _ => false @[simp] theorem isEmpty_nil : ([] : List α).isEmpty = true := rfl @[simp] theorem isEmpty_cons : (x :: xs : List α).isEmpty = false := rfl /-! ### elem -/ /-- `O(|l|)`. `elem a l` or `l.contains a` is true if there is an element in `l` equal to `a`. * `elem 3 [1, 4, 2, 3, 3, 7] = true` * `elem 5 [1, 4, 2, 3, 3, 7] = false` -/ def elem [BEq α] (a : α) : List α → Bool | [] => false | b::bs => match a == b with | true => true | false => elem a bs @[simp] theorem elem_nil [BEq α] : ([] : List α).elem a = false := rfl theorem elem_cons [BEq α] {a : α} : (b::bs).elem a = match a == b with | true => true | false => bs.elem a := rfl /-- `notElem a l` is `!(elem a l)`. -/ @[deprecated (since := "2024-06-15")] def notElem [BEq α] (a : α) (as : List α) : Bool := !(as.elem a) /-! ### contains -/ @[inherit_doc elem] abbrev contains [BEq α] (as : List α) (a : α) : Bool := elem a as @[simp] theorem contains_nil [BEq α] : ([] : List α).contains a = false := rfl /-! ### 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` -/ inductive Mem (a : α) : List α → Prop /-- The head of a list is a member: `a ∈ a :: as`. -/ | head (as : List α) : Mem a (a::as) /-- A member of the tail of a list is a member of the list: `a ∈ l → a ∈ b :: l`. -/ | tail (b : α) {as : List α} : Mem a as → Mem a (b::as) instance : Membership α (List α) where mem := Mem theorem mem_of_elem_eq_true [BEq α] [LawfulBEq α] {a : α} {as : List α} : elem a as = true → a ∈ as := by match as with | [] => simp [elem] | a'::as => simp [elem] split next h => intros; simp [BEq.beq] at h; subst h; apply Mem.head next _ => intro h; exact Mem.tail _ (mem_of_elem_eq_true h) theorem elem_eq_true_of_mem [BEq α] [LawfulBEq α] {a : α} {as : List α} (h : a ∈ as) : elem a as = true := by induction h with | head _ => simp [elem] | tail _ _ ih => simp [elem]; split; rfl; assumption instance [BEq α] [LawfulBEq α] (a : α) (as : List α) : Decidable (a ∈ as) := decidable_of_decidable_of_iff (Iff.intro mem_of_elem_eq_true elem_eq_true_of_mem) theorem mem_append_of_mem_left {a : α} {as : List α} (bs : List α) : a ∈ as → a ∈ as ++ bs := by intro h induction h with | head => apply Mem.head | tail => apply Mem.tail; assumption theorem mem_append_of_mem_right {b : α} {bs : List α} (as : List α) : b ∈ bs → b ∈ as ++ bs := by intro h induction as with | nil => simp [h] | cons => apply Mem.tail; assumption instance decidableBEx (p : α → Prop) [DecidablePred p] : ∀ l : List α, Decidable (Exists fun x => x ∈ l ∧ p x) | [] => isFalse nofun | x :: xs => if h₁ : p x then isTrue ⟨x, .head .., h₁⟩ else match decidableBEx p xs with | isTrue h₂ => isTrue <| let ⟨y, hm, hp⟩ := h₂; ⟨y, .tail _ hm, hp⟩ | isFalse h₂ => isFalse fun | ⟨y, .tail _ h, hp⟩ => h₂ ⟨y, h, hp⟩ | ⟨_, .head .., hp⟩ => h₁ hp instance decidableBAll (p : α → Prop) [DecidablePred p] : ∀ l : List α, Decidable (∀ x, x ∈ l → p x) | [] => isTrue nofun | x :: xs => if h₁ : p x then match decidableBAll p xs with | isTrue h₂ => isTrue fun | y, .tail _ h => h₂ y h | _, .head .. => h₁ | isFalse h₂ => isFalse fun H => h₂ fun y hm => H y (.tail _ hm) else isFalse fun H => h₁ <| H x (.head ..) /-! ## Sublists -/ /-! ### take -/ /-- `O(min n |xs|)`. Returns the first `n` elements of `xs`, or the whole list if `n` is too large. * `take 0 [a, b, c, d, e] = []` * `take 3 [a, b, c, d, e] = [a, b, c]` * `take 6 [a, b, c, d, e] = [a, b, c, d, e]` -/ def take : Nat → List α → List α | 0, _ => [] | _+1, [] => [] | n+1, a::as => a :: take n as @[simp] theorem take_nil : ([] : List α).take i = [] := by cases i <;> rfl @[simp] theorem take_zero (l : List α) : l.take 0 = [] := rfl @[simp] theorem take_cons_succ : (a::as).take (i+1) = a :: as.take i := rfl /-! ### drop -/ /-- `O(min n |xs|)`. Removes the first `n` elements of `xs`. * `drop 0 [a, b, c, d, e] = [a, b, c, d, e]` * `drop 3 [a, b, c, d, e] = [d, e]` * `drop 6 [a, b, c, d, e] = []` -/ def drop : Nat → List α → List α | 0, a => a | _+1, [] => [] | n+1, _::as => drop n as @[simp] theorem drop_nil : ([] : List α).drop i = [] := by cases i <;> rfl @[simp] theorem drop_zero (l : List α) : l.drop 0 = l := rfl @[simp] theorem drop_succ_cons : (a :: l).drop (n + 1) = l.drop n := rfl theorem drop_eq_nil_of_le {as : List α} {i : Nat} (h : as.length ≤ i) : as.drop i = [] := by match as, i with | [], i => simp | _::_, 0 => simp at h | _::as, i+1 => simp only [length_cons] at h; exact @drop_eq_nil_of_le as i (Nat.le_of_succ_le_succ h) /-! ### takeWhile -/ /-- `O(|xs|)`. Returns the longest initial segment of `xs` for which `p` returns true. * `takeWhile (· > 5) [7, 6, 4, 8] = [7, 6]` * `takeWhile (· > 5) [7, 6, 6, 8] = [7, 6, 6, 8]` -/ def takeWhile (p : α → Bool) : (xs : List α) → List α | [] => [] | hd :: tl => match p hd with | true => hd :: takeWhile p tl | false => [] @[simp] theorem takeWhile_nil : ([] : List α).takeWhile p = [] := rfl /-! ### 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] ``` -/ def dropWhile (p : α → Bool) : List α → List α | [] => [] | a::l => match p a with | true => dropWhile p l | false => a::l @[simp] theorem dropWhile_nil : ([] : List α).dropWhile p = [] := rfl /-! ### 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]) ``` -/ @[inline] def partition (p : α → Bool) (as : List α) : List α × List α := loop as ([], []) where @[specialize] loop : List α → List α × List α → List α × List α | [], (bs, cs) => (bs.reverse, cs.reverse) | a::as, (bs, cs) => match p a with | true => loop as (a::bs, cs) | false => loop as (bs, a::cs) /-! ### dropLast -/ /-- Removes the last element of the list. * `dropLast [] = []` * `dropLast [a] = []` * `dropLast [a, b, c] = [a, b]` -/ def dropLast {α} : List α → List α | [] => [] | [_] => [] | a::as => a :: dropLast as @[simp] theorem dropLast_nil : ([] : List α).dropLast = [] := rfl @[simp] theorem dropLast_single : [x].dropLast = [] := rfl @[simp] theorem dropLast_cons₂ : (x::y::zs).dropLast = x :: (y::zs).dropLast := rfl -- Later this can be proved by `simp` via `[List.length_dropLast, List.length_cons, Nat.add_sub_cancel]`, -- but we need this while bootstrapping `Array`. @[simp] theorem length_dropLast_cons (a : α) (as : List α) : (a :: as).dropLast.length = as.length := by match as with | [] => rfl | b::bs => have ih := length_dropLast_cons b bs simp [dropLast, ih] /-! ### isPrefixOf -/ /-- `isPrefixOf l₁ l₂` returns `true` Iff `l₁` is a prefix of `l₂`. That is, there exists a `t` such that `l₂ == l₁ ++ t`. -/ def isPrefixOf [BEq α] : List α → List α → Bool | [], _ => true | _, [] => false | a::as, b::bs => a == b && isPrefixOf as bs @[simp] theorem isPrefixOf_nil_left [BEq α] : isPrefixOf ([] : List α) l = true := by simp [isPrefixOf] @[simp] theorem isPrefixOf_cons_nil [BEq α] : isPrefixOf (a::as) ([] : List α) = false := rfl theorem isPrefixOf_cons₂ [BEq α] {a : α} : isPrefixOf (a::as) (b::bs) = (a == b && isPrefixOf as bs) := rfl /-! ### isPrefixOf? -/ /-- `isPrefixOf? l₁ l₂` returns `some t` when `l₂ == l₁ ++ t`. -/ def isPrefixOf? [BEq α] : List α → List α → Option (List α) | [], l₂ => some l₂ | _, [] => none | (x₁ :: l₁), (x₂ :: l₂) => if x₁ == x₂ then isPrefixOf? l₁ l₂ else none /-! ### isSuffixOf -/ /-- `isSuffixOf l₁ l₂` returns `true` Iff `l₁` is a suffix of `l₂`. That is, there exists a `t` such that `l₂ == t ++ l₁`. -/ def isSuffixOf [BEq α] (l₁ l₂ : List α) : Bool := isPrefixOf l₁.reverse l₂.reverse @[simp] theorem isSuffixOf_nil_left [BEq α] : isSuffixOf ([] : List α) l = true := by simp [isSuffixOf] /-! ### isSuffixOf? -/ /-- `isSuffixOf? l₁ l₂` returns `some t` when `l₂ == t ++ l₁`.-/ def isSuffixOf? [BEq α] (l₁ l₂ : List α) : Option (List α) := Option.map List.reverse <| isPrefixOf? l₁.reverse l₂.reverse /-! ### Subset -/ /-- `l₁ ⊆ l₂` means that every element of `l₁` is also an element of `l₂`, ignoring multiplicity. -/ protected def Subset (l₁ l₂ : List α) := ∀ ⦃a : α⦄, a ∈ l₁ → a ∈ l₂ instance : HasSubset (List α) := ⟨List.Subset⟩ instance [DecidableEq α] : DecidableRel (Subset : List α → List α → Prop) := fun _ _ => decidableBAll _ _ /-! ### Sublist and isSublist -/ /-- `l₁ <+ l₂`, or `Sublist l₁ l₂`, says that `l₁` is a (non-contiguous) subsequence of `l₂`. -/ inductive Sublist {α} : List α → List α → Prop /-- the base case: `[]` is a sublist of `[]` -/ | slnil : Sublist [] [] /-- If `l₁` is a subsequence of `l₂`, then it is also a subsequence of `a :: l₂`. -/ | cons a : Sublist l₁ l₂ → Sublist l₁ (a :: l₂) /-- If `l₁` is a subsequence of `l₂`, then `a :: l₁` is a subsequence of `a :: l₂`. -/ | cons₂ a : Sublist l₁ l₂ → Sublist (a :: l₁) (a :: l₂) @[inherit_doc] scoped infixl:50 " <+ " => Sublist /-- True if the first list is a potentially non-contiguous sub-sequence of the second list. -/ def isSublist [BEq α] : List α → List α → Bool | [], _ => true | _, [] => false | l₁@(hd₁::tl₁), hd₂::tl₂ => if hd₁ == hd₂ then tl₁.isSublist tl₂ else l₁.isSublist tl₂ /-! ### 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]` -/ def rotateLeft (xs : List α) (n : Nat := 1) : List α := let len := xs.length if len ≤ 1 then xs else let n := n % len let b := xs.take n let e := xs.drop n e ++ b @[simp] theorem rotateLeft_nil : ([] : List α).rotateLeft n = [] := rfl /-! ### 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]` -/ def rotateRight (xs : List α) (n : Nat := 1) : List α := let len := xs.length if len ≤ 1 then xs else let n := len - n % len let b := xs.take n let e := xs.drop n e ++ b @[simp] theorem rotateRight_nil : ([] : List α).rotateRight n = [] := rfl /-! ## Manipulating elements -/ /-! ### replace -/ /-- `O(|l|)`. `replace l a b` replaces the first element in the list equal to `a` with `b`. * `replace [1, 4, 2, 3, 3, 7] 3 6 = [1, 4, 2, 6, 3, 7]` * `replace [1, 4, 2, 3, 3, 7] 5 6 = [1, 4, 2, 3, 3, 7]` -/ def replace [BEq α] : List α → α → α → List α | [], _, _ => [] | a::as, b, c => match b == a with | true => c::as | false => a :: replace as b c @[simp] theorem replace_nil [BEq α] : ([] : List α).replace a b = [] := rfl theorem replace_cons [BEq α] {a : α} : (a::as).replace b c = match b == a with | true => c::as | false => a :: replace as b c := rfl /-! ### insert -/ /-- Inserts an element into a list without duplication. -/ @[inline] protected def insert [BEq α] (a : α) (l : List α) : List α := if l.elem a then l else a :: l /-! ### erase -/ /-- `O(|l|)`. `erase l a` removes the first occurrence of `a` from `l`. * `erase [1, 5, 3, 2, 5] 5 = [1, 3, 2, 5]` * `erase [1, 5, 3, 2, 5] 6 = [1, 5, 3, 2, 5]` -/ protected def erase {α} [BEq α] : List α → α → List α | [], _ => [] | a::as, b => match a == b with | true => as | false => a :: List.erase as b @[simp] theorem erase_nil [BEq α] (a : α) : [].erase a = [] := rfl theorem erase_cons [BEq α] (a b : α) (l : List α) : (b :: l).erase a = if b == a then l else b :: l.erase a := by simp only [List.erase]; split <;> simp_all /-! ### eraseIdx -/ /-- `O(i)`. `eraseIdx l i` removes the `i`'th element of the list `l`. * `erase [a, b, c, d, e] 0 = [b, c, d, e]` * `erase [a, b, c, d, e] 1 = [a, c, d, e]` * `erase [a, b, c, d, e] 5 = [a, b, c, d, e]` -/ def eraseIdx : List α → Nat → List α | [], _ => [] | _::as, 0 => as | a::as, n+1 => a :: eraseIdx as n @[simp] theorem eraseIdx_nil : ([] : List α).eraseIdx i = [] := rfl @[simp] theorem eraseIdx_cons_zero : (a::as).eraseIdx 0 = as := rfl @[simp] theorem eraseIdx_cons_succ : (a::as).eraseIdx (i+1) = a :: as.eraseIdx i := rfl /-! ### find? -/ /-- `O(|l|)`. `find? p l` returns the first element for which `p` returns true, or `none` if no such element is found. * `find? (· < 5) [7, 6, 5, 8, 1, 2, 6] = some 1` * `find? (· < 1) [7, 6, 5, 8, 1, 2, 6] = none` -/ def find? (p : α → Bool) : List α → Option α | [] => none | a::as => match p a with | true => some a | false => find? p as @[simp] theorem find?_nil : ([] : List α).find? p = none := rfl theorem find?_cons : (a::as).find? p = match p a with | true => some a | false => as.find? p := rfl /-! ### 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` -/ def findSome? (f : α → Option β) : List α → Option β | [] => none | a::as => match f a with | some b => some b | none => findSome? f as @[simp] theorem findSome?_nil : ([] : List α).findSome? f = none := rfl theorem findSome?_cons {f : α → Option β} : (a::as).findSome? f = match f a with | some b => some b | none => as.findSome? f := rfl /-! ### 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`. * `lookup 3 [(1, 2), (3, 4), (3, 5)] = some 4` * `lookup 2 [(1, 2), (3, 4), (3, 5)] = none` -/ def lookup [BEq α] : α → List (α × β) → Option β | _, [] => none | a, (k,b)::es => match a == k with | true => some b | false => lookup a es @[simp] theorem lookup_nil [BEq α] : ([] : List (α × β)).lookup a = none := rfl theorem lookup_cons [BEq α] {k : α} : ((k,b)::es).lookup a = match a == k with | true => some b | false => es.lookup a := rfl /-! ## Logical operations -/ /-! ### any -/ /-- `O(|l|)`. Returns true if `p` is true for any element of `l`. * `any p [a, b, c] = p a || p b || p c` Short-circuits upon encountering the first `true`. -/ def any : List α → (α → Bool) → Bool | [], _ => false | h :: t, p => p h || any t p @[simp] theorem any_nil : [].any f = false := rfl @[simp] theorem any_cons : (a::l).any f = (f a || l.any f) := rfl /-! ### all -/ /-- `O(|l|)`. Returns true if `p` is true for every element of `l`. * `all p [a, b, c] = p a && p b && p c` Short-circuits upon encountering the first `false`. -/ def all : List α → (α → Bool) → Bool | [], _ => true | h :: t, p => p h && all t p @[simp] theorem all_nil : [].all f = true := rfl @[simp] theorem all_cons : (a::l).all f = (f a && l.all f) := rfl /-! ### or -/ /-- `O(|l|)`. Returns true if `true` is an element of the list of booleans `l`. * `or [a, b, c] = a || b || c` -/ def or (bs : List Bool) : Bool := bs.any id @[simp] theorem or_nil : [].or = false := rfl @[simp] theorem or_cons : (a::l).or = (a || l.or) := rfl /-! ### and -/ /-- `O(|l|)`. Returns true if every element of `l` is the value `true`. * `and [a, b, c] = a && b && c` -/ def and (bs : List Bool) : Bool := bs.all id @[simp] theorem and_nil : [].and = true := rfl @[simp] theorem and_cons : (a::l).and = (a && l.and) := rfl /-! ## 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₃]` -/ @[specialize] def zipWith (f : α → β → γ) : (xs : List α) → (ys : List β) → List γ | x::xs, y::ys => f x y :: zipWith f xs ys | _, _ => [] @[simp] theorem zipWith_nil_left {f : α → β → γ} : zipWith f [] l = [] := rfl @[simp] theorem zipWith_nil_right {f : α → β → γ} : zipWith f l [] = [] := by simp [zipWith] @[simp] theorem zipWith_cons_cons {f : α → β → γ} : zipWith f (a :: as) (b :: bs) = f a b :: zipWith f as bs := rfl /-! ### 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₃)]` -/ def zip : List α → List β → List (Prod α β) := zipWith Prod.mk @[simp] theorem zip_nil_left : zip ([] : List α) (l : List β) = [] := rfl @[simp] theorem zip_nil_right : zip (l : List α) ([] : List β) = [] := by simp [zip, zipWith] @[simp] theorem zip_cons_cons : zip (a :: as) (b :: bs) = (a, b) :: zip as bs := rfl /-! ### 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. -/ def zipWithAll (f : Option α → Option β → γ) : List α → List β → List γ | [], bs => bs.map fun b => f none (some b) | a :: as, [] => (a :: as).map fun a => f (some a) none | a :: as, b :: bs => f a b :: zipWithAll f as bs @[simp] theorem zipWithAll_nil_right : zipWithAll f as [] = as.map fun a => f (some a) none := by cases as <;> rfl @[simp] theorem zipWithAll_nil_left : zipWithAll f [] bs = bs.map fun b => f none (some b) := rfl @[simp] theorem zipWithAll_cons_cons : zipWithAll f (a :: as) (b :: bs) = f (some a) (some b) :: zipWithAll f as bs := rfl /-! ### 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₃])` -/ def unzip : List (α × β) → List α × List β | [] => ([], []) | (a, b) :: t => match unzip t with | (al, bl) => (a::al, b::bl) @[simp] theorem unzip_nil : ([] : List (α × β)).unzip = ([], []) := rfl @[simp] theorem unzip_cons {h : α × β} : (h :: t).unzip = match unzip t with | (al, bl) => (h.1::al, h.2::bl) := rfl /-! ## 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]` -/ def range (n : Nat) : List Nat := loop n [] where loop : Nat → List Nat → List Nat | 0, ns => ns | n+1, ns => loop n (n::ns) @[simp] theorem range_zero : range 0 = [] := rfl /-! ### iota -/ /-- `O(n)`. `iota n` is the numbers from `1` to `n` inclusive, in decreasing order. * `iota 5 = [5, 4, 3, 2, 1]` -/ def iota : Nat → List Nat | 0 => [] | m@(n+1) => m :: iota n @[simp] theorem iota_zero : iota 0 = [] := rfl @[simp] theorem iota_succ : iota (i+1) = (i+1) :: iota i := rfl /-! ### 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)]` -/ def enumFrom : Nat → List α → List (Nat × α) | _, [] => nil | n, x :: xs => (n, x) :: enumFrom (n + 1) xs @[simp] theorem enumFrom_nil : ([] : List α).enumFrom i = [] := rfl @[simp] theorem enumFrom_cons : (a::as).enumFrom i = (i, a) :: as.enumFrom (i+1) := rfl /-! ### enum -/ /-- `O(|l|)`. `enum l` pairs up each element with its index in the list. * `enum [a, b, c] = [(0, a), (1, b), (2, c)]` -/ def enum : List α → List (Nat × α) := enumFrom 0 @[simp] theorem enum_nil : ([] : List α).enum = [] := rfl /-! ## Minima and maxima -/ /-! ### minimum? -/ /-- Returns the smallest element of the list, if it is not empty. * `[].minimum? = none` * `[4].minimum? = some 4` * `[1, 4, 2, 10, 6].minimum? = some 1` -/ def minimum? [Min α] : List α → Option α | [] => none | a::as => some <| as.foldl min a /-! ### maximum? -/ /-- Returns the largest element of the list, if it is not empty. * `[].maximum? = none` * `[4].maximum? = some 4` * `[1, 4, 2, 10, 6].maximum? = some 10` -/ def maximum? [Max α] : List α → Option α | [] => none | a::as => some <| as.foldl max a /-! ## 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]` -/ def intersperse (sep : α) : List α → List α | [] => [] | [x] => [x] | x::xs => x :: sep :: intersperse sep xs @[simp] theorem intersperse_nil (sep : α) : ([] : List α).intersperse sep = [] := rfl @[simp] theorem intersperse_single (sep : α) : [x].intersperse sep = [x] := rfl @[simp] theorem intersperse_cons₂ (sep : α) : (x::y::zs).intersperse sep = x::sep::((y::zs).intersperse sep) := rfl /-! ### 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` -/ def intercalate (sep : List α) (xs : List (List α)) : List α := join (intersperse sep xs) /-! ### 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]` -/ def eraseDups {α} [BEq α] (as : List α) : List α := loop as [] where loop : List α → List α → List α | [], bs => bs.reverse | a::as, bs => match bs.elem a with | true => loop as bs | false => loop as (a::bs) /-! ### eraseReps -/ /-- `O(|l|)`. Erase repeated adjacent elements. Keeps the first occurrence of each run. * `eraseReps [1, 3, 2, 2, 2, 3, 5] = [1, 3, 2, 3, 5]` -/ def eraseReps {α} [BEq α] : List α → List α | [] => [] | a::as => loop a as [] where loop {α} [BEq α] : α → List α → List α → List α | a, [], rs => (a::rs).reverse | a, a'::as, rs => match a == a' with | true => loop a as rs | false => loop a' as (a::rs) /-! ### 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])` -/ @[inline] def span (p : α → Bool) (as : List α) : List α × List α := loop as [] where @[specialize] loop : List α → List α → List α × List α | [], rs => (rs.reverse, []) | a::as, rs => match p a with | true => loop as (a::rs) | false => (rs.reverse, a::as) /-! ### groupBy -/ /-- `O(|l|)`. `groupBy R l` splits `l` into chains of elements such that adjacent elements are related by `R`. * `groupBy (·==·) [1, 1, 2, 2, 2, 3, 2] = [[1, 1], [2, 2, 2], [3], [2]]` * `groupBy (·<·) [1, 2, 5, 4, 5, 1, 4] = [[1, 2, 5], [4, 5], [1, 4]]` -/ @[specialize] def groupBy (R : α → α → Bool) : List α → List (List α) | [] => [] | a::as => loop as a [] [] where @[specialize] loop : List α → α → List α → List (List α) → List (List α) | a::as, ag, g, gs => match R ag a with | true => loop as a (ag::g) gs | false => loop as a [] ((ag::g).reverse::gs) | [], ag, g, gs => ((ag::g).reverse::gs).reverse /-! ### 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]` -/ def removeAll [BEq α] (xs ys : List α) : List α := xs.filter (fun x => !ys.elem x) end List