392 lines
14 KiB
Text
392 lines
14 KiB
Text
/-
|
||
Copyright (c) 2014 Parikshit Khanna. All rights reserved.
|
||
Released under Apache 2.0 license as described in the file LICENSE.
|
||
Author: Parikshit Khanna, Jeremy Avigad, Leonardo de Moura, Floris van Doorn
|
||
-/
|
||
prelude
|
||
import init.data.list.basic init.function init.meta init.data.nat.lemmas
|
||
import init.meta.interactive init.meta.smt.rsimp
|
||
|
||
universes u v w
|
||
variables {α : Type u} {β : Type v} {γ : Type w}
|
||
|
||
namespace list
|
||
open nat
|
||
|
||
/- append -/
|
||
|
||
@[simp] lemma nil_append (s : list α) : [] ++ s = s :=
|
||
rfl
|
||
|
||
@[simp] lemma cons_append (x : α) (s t : list α) : (x::s) ++ t = x::(s ++ t) :=
|
||
rfl
|
||
|
||
@[simp] lemma append_nil (t : list α) : t ++ [] = t :=
|
||
by induction t; simph
|
||
|
||
@[simp] lemma append_assoc (s t u : list α) : s ++ t ++ u = s ++ (t ++ u) :=
|
||
by induction s; simph
|
||
|
||
lemma append_ne_nil_of_ne_nil_left (s t : list α) : s ≠ [] → s ++ t ≠ nil :=
|
||
by induction s; intros; contradiction
|
||
|
||
lemma append_ne_nil_of_ne_nil_right (s t : list α) : t ≠ [] → s ++ t ≠ nil :=
|
||
by induction s; intros; contradiction
|
||
|
||
/- concat -/
|
||
|
||
@[simp] lemma concat_eq_append (a : α) (l : list α) : concat l a = l ++ [a] :=
|
||
by induction l; simph [concat]
|
||
|
||
/- head & tail -/
|
||
|
||
attribute [simp] head tail
|
||
|
||
@[simp] lemma head_append [h : inhabited α] (t : list α) {s : list α} (h : s ≠ []) : head (s ++ t) = head s :=
|
||
by {induction s, contradiction, simp}
|
||
|
||
lemma cons_head_tail [h : inhabited α] {l : list α} (h : l ≠ []) : (head l)::(tail l) = l :=
|
||
by {induction l, contradiction, simp}
|
||
|
||
/- repeat taken dropn -/
|
||
|
||
attribute [simp] repeat taken dropn
|
||
|
||
/- length -/
|
||
|
||
attribute [simp] length
|
||
|
||
lemma length_cons (a : α) (l : list α) : length (a :: l) = length l + 1 :=
|
||
rfl
|
||
|
||
@[simp] lemma length_append (s t : list α) : length (s ++ t) = length s + length t :=
|
||
by induction s; simph
|
||
|
||
lemma length_concat (a : α) (l : list α) : length (concat l a) = succ (length l) :=
|
||
by simp
|
||
|
||
@[simp] lemma length_repeat (a : α) (n : ℕ) : length (repeat a n) = n :=
|
||
by induction n; simph
|
||
|
||
lemma eq_nil_of_length_eq_zero {l : list α} : length l = 0 → l = [] :=
|
||
by {induction l; intros, refl, contradiction}
|
||
|
||
lemma ne_nil_of_length_eq_succ {l : list α} : ∀ {n : nat}, length l = succ n → l ≠ [] :=
|
||
by induction l; intros; contradiction
|
||
|
||
-- TODO(Leo): cleanup proof after arith dec proc
|
||
@[simp] lemma length_taken : ∀ (i : ℕ) (l : list α), length (taken i l) = min i (length l)
|
||
| 0 l := by simp
|
||
| (succ n) [] := by simp
|
||
| (succ n) (a::l) := by simph [nat.min_succ_succ]; rw [-add_one_eq_succ]; simp
|
||
|
||
-- TODO(Leo): cleanup proof after arith dec proc
|
||
@[simp] lemma length_dropn : ∀ (i : ℕ) (l : list α), length (dropn i l) = length l - i
|
||
| 0 l := rfl
|
||
| (succ i) [] := eq.symm (nat.zero_sub_eq_zero (succ i))
|
||
| (succ i) (x::l) := calc
|
||
length (dropn (succ i) (x::l))
|
||
= length l - i : length_dropn i l
|
||
... = succ (length l) - succ i : nat.sub_eq_succ_sub_succ (length l) i
|
||
|
||
/- map -/
|
||
|
||
attribute [simp] map
|
||
|
||
lemma map_cons (f : α → β) (a l) : map f (a::l) = f a :: map f l :=
|
||
rfl
|
||
|
||
@[simp] lemma map_append (f : α → β) : ∀ l₁ l₂, map f (l₁ ++ l₂) = (map f l₁) ++ (map f l₂) :=
|
||
by intro l₁; induction l₁; intros; simph
|
||
|
||
lemma map_singleton (f : α → β) (a : α) : map f [a] = [f a] :=
|
||
rfl
|
||
|
||
lemma map_concat (f : α → β) (a : α) (l : list α) : map f (concat l a) = concat (map f l) (f a) :=
|
||
by induction l; simph
|
||
|
||
@[simp] lemma map_id (l : list α) : map id l = l :=
|
||
by induction l; simph
|
||
|
||
lemma map_id' {f : α → α} (h : ∀ x, f x = x) (l : list α) : map f l = l :=
|
||
by induction l; simph
|
||
|
||
@[simp] lemma map_map (g : β → γ) (f : α → β) (l : list α) : map g (map f l) = map (g ∘ f) l :=
|
||
by induction l; simph
|
||
|
||
@[simp] lemma length_map (f : α → β) (l : list α) : length (map f l) = length l :=
|
||
by induction l; simph
|
||
|
||
/- map₂ -/
|
||
|
||
attribute [simp] map₂
|
||
|
||
@[simp] lemma length_map₂ (f : α → β → γ) (l₁) : ∀ l₂, length (map₂ f l₁ l₂) = min (length l₁) (length l₂) :=
|
||
by {induction l₁; intro l₂; cases l₂; simph [add_one_eq_succ, min_succ_succ]}
|
||
|
||
/- reverse -/
|
||
|
||
@[simp] lemma reverse_nil : reverse (@nil α) = [] :=
|
||
rfl
|
||
|
||
local attribute [simp] reverse_core
|
||
|
||
@[simp] lemma reverse_cons (a : α) (l : list α) : reverse (a::l) = concat (reverse l) a :=
|
||
have aux : ∀ l₁ l₂, reverse_core l₁ (concat l₂ a) = concat (reverse_core l₁ l₂) a,
|
||
by intros l₁; induction l₁; intros; rsimp,
|
||
aux l nil
|
||
|
||
@[simp] lemma reverse_singleton (a : α) : reverse [a] = [a] :=
|
||
rfl
|
||
|
||
@[simp] lemma reverse_append (s t : list α) : reverse (s ++ t) = (reverse t) ++ (reverse s) :=
|
||
by induction s; simph
|
||
|
||
@[simp] lemma reverse_reverse (l : list α) : reverse (reverse l) = l :=
|
||
by induction l; simph
|
||
|
||
lemma concat_eq_reverse_cons (a : α) (l : list α) : concat l a = reverse (a :: reverse l) :=
|
||
by induction l; simph
|
||
|
||
@[simp] lemma length_reverse (l : list α) : length (reverse l) = length l :=
|
||
by induction l; simph
|
||
|
||
@[simp] lemma map_reverse (f : α → β) (l : list α) : map f (reverse l) = reverse (map f l) :=
|
||
by induction l; simph
|
||
|
||
/- last -/
|
||
|
||
attribute [simp] last
|
||
|
||
@[simp] lemma last_cons {a : α} {l : list α} : ∀ (h₁ : a :: l ≠ nil) (h₂ : l ≠ nil), last (a :: l) h₁ = last l h₂ :=
|
||
by {induction l; intros, contradiction, simph}
|
||
|
||
@[simp] lemma last_append {a : α} (l : list α) (h : l ++ [a] ≠ []) : last (l ++ [a]) h = a :=
|
||
begin
|
||
induction l with hd tl ih; rsimp,
|
||
assert haux : tl ++ [a] ≠ [],
|
||
{apply append_ne_nil_of_ne_nil_right, contradiction},
|
||
simph
|
||
end
|
||
|
||
lemma last_concat {a : α} (l : list α) (h : concat l a ≠ []) : last (concat l a) h = a :=
|
||
by simph
|
||
|
||
/- nth -/
|
||
|
||
attribute [simp] nth
|
||
|
||
lemma nth_succ (a : α) (l : list α) (n : nat) : nth (a::l) (nat.succ n) = nth l n :=
|
||
rfl
|
||
|
||
/- mem -/
|
||
|
||
@[simp] lemma mem_nil_iff (a : α) : a ∈ ([] : list α) ↔ false :=
|
||
iff.rfl
|
||
|
||
@[simp] lemma not_mem_nil (a : α) : a ∉ ([] : list α) :=
|
||
iff.mp $ mem_nil_iff a
|
||
|
||
@[simp] lemma mem_cons_self (a : α) (l : list α) : a ∈ a :: l :=
|
||
or.inl rfl
|
||
|
||
@[simp] lemma mem_cons_iff (a y : α) (l : list α) : a ∈ y :: l ↔ (a = y ∨ a ∈ l) :=
|
||
iff.rfl
|
||
|
||
@[rsimp] lemma mem_cons_eq (a y : α) (l : list α) : (a ∈ y :: l) = (a = y ∨ a ∈ l) :=
|
||
rfl
|
||
|
||
lemma eq_nil_of_forall_not_mem : ∀ {l : list α}, (∀ a, a ∉ l) → l = nil
|
||
| [] := assume h, rfl
|
||
| (b :: l') := assume h, absurd (mem_cons_self b l') (h b)
|
||
|
||
lemma mem_cons_of_mem (y : α) {a : α} {l : list α} : a ∈ l → a ∈ y :: l :=
|
||
assume H, or.inr H
|
||
|
||
lemma eq_or_mem_of_mem_cons {a y : α} {l : list α} : a ∈ y::l → a = y ∨ a ∈ l :=
|
||
assume h, h
|
||
|
||
lemma mem_singleton {a b : α} : a ∈ [b] → a = b :=
|
||
suppose a ∈ [b], or.elim (eq_or_mem_of_mem_cons this)
|
||
(suppose a = b, this)
|
||
(suppose a ∈ [], absurd this (not_mem_nil a))
|
||
|
||
@[simp] lemma mem_singleton_iff (a b : α) : a ∈ [b] ↔ a = b :=
|
||
iff.intro mem_singleton (begin intro h, simp [h] end)
|
||
|
||
lemma mem_of_mem_cons_of_mem {a b : α} {l : list α} : a ∈ b::l → b ∈ l → a ∈ l :=
|
||
assume ainbl binl, or.elim (eq_or_mem_of_mem_cons ainbl)
|
||
(suppose a = b, begin subst a, exact binl end)
|
||
(suppose a ∈ l, this)
|
||
|
||
lemma mem_or_mem_of_mem_append {a : α} : ∀ {s t : list α}, a ∈ s ++ t → a ∈ s ∨ a ∈ t
|
||
| [] t h := or.inr h
|
||
| (b::s) t h :=
|
||
have a = b ∨ a ∈ s ++ t, from h,
|
||
match this with
|
||
| or.inl h₁ := or.inl (h₁ ▸ mem_cons_self _ _)
|
||
| or.inr h₁ :=
|
||
have a ∈ s ∨ a ∈ t, from mem_or_mem_of_mem_append h₁,
|
||
match this with
|
||
| or.inl h₂ := or.inl (mem_cons_of_mem _ h₂)
|
||
| or.inr h₂ := or.inr h₂
|
||
end
|
||
end
|
||
|
||
lemma mem_append_of_mem_or_mem {a : α} {s t : list α} : a ∈ s ∨ a ∈ t → a ∈ s ++ t :=
|
||
list.rec_on s
|
||
(take h, or.elim h false.elim id)
|
||
(take b s,
|
||
assume ih : a ∈ s ∨ a ∈ t → a ∈ s ++ t,
|
||
suppose a ∈ b::s ∨ a ∈ t,
|
||
or.elim this
|
||
(suppose a ∈ b::s,
|
||
or.elim (eq_or_mem_of_mem_cons this)
|
||
(suppose a = b, or.inl this)
|
||
(suppose a ∈ s, or.inr (ih (or.inl this))))
|
||
(suppose a ∈ t, or.inr (ih (or.inr this))))
|
||
|
||
@[simp] lemma mem_append_iff (a : α) (s t : list α) : a ∈ s ++ t ↔ a ∈ s ∨ a ∈ t :=
|
||
iff.intro mem_or_mem_of_mem_append mem_append_of_mem_or_mem
|
||
|
||
@[rsimp] lemma mem_append_eq (a : α) (s t : list α) : (a ∈ s ++ t) = (a ∈ s ∨ a ∈ t) :=
|
||
propext $ mem_append_iff a s t
|
||
|
||
lemma not_mem_of_not_mem_append_left {a : α} {s t : list α} : a ∉ s++t → a ∉ s :=
|
||
λ nainst ains, absurd (mem_append_of_mem_or_mem (or.inl ains)) nainst
|
||
|
||
lemma not_mem_of_not_mem_append_right {a : α} {s t : list α} : a ∉ s++t → a ∉ t :=
|
||
λ nainst aint, absurd (mem_append_of_mem_or_mem (or.inr aint)) nainst
|
||
|
||
lemma not_mem_append {a : α} {s t : list α} : a ∉ s → a ∉ t → a ∉ s++t :=
|
||
λ nains naint ainst, or.elim (mem_or_mem_of_mem_append ainst)
|
||
(λ ains, by contradiction)
|
||
(λ aint, by contradiction)
|
||
|
||
lemma length_pos_of_mem {a : α} : ∀ {l : list α}, a ∈ l → 0 < length l
|
||
| [] := suppose a ∈ [], begin simp at this, contradiction end
|
||
| (b::l) := suppose a ∈ b::l, nat.zero_lt_succ _
|
||
|
||
lemma mem_split {a : α} {l : list α} : a ∈ l → ∃ s t : list α, l = s ++ (a::t) :=
|
||
list.rec_on l
|
||
(suppose a ∈ [], begin simp at this, contradiction end)
|
||
(take b l,
|
||
assume ih : a ∈ l → ∃ s t : list α, l = s ++ (a::t),
|
||
suppose a ∈ b::l,
|
||
or.elim (eq_or_mem_of_mem_cons this)
|
||
(suppose a = b, ⟨[], l, begin rw this, reflexivity end⟩)
|
||
(suppose a ∈ l,
|
||
match (ih this) with
|
||
| ⟨s, t, (h : l = s ++ (a::t))⟩ := ⟨b::s, t, begin rw h, reflexivity end⟩
|
||
end))
|
||
|
||
lemma mem_append_left {a : α} {l₁ : list α} (l₂ : list α) : a ∈ l₁ → a ∈ l₁ ++ l₂ :=
|
||
assume ainl₁, mem_append_of_mem_or_mem (or.inl ainl₁)
|
||
|
||
lemma mem_append_right {a : α} (l₁ : list α) {l₂ : list α} : a ∈ l₂ → a ∈ l₁ ++ l₂ :=
|
||
assume ainl₂, mem_append_of_mem_or_mem (or.inr ainl₂)
|
||
|
||
lemma mem_of_ne_of_mem {a y : α} {l : list α} (h₁ : a ≠ y) (h₂ : a ∈ y :: l) : a ∈ l :=
|
||
or.elim (eq_or_mem_of_mem_cons h₂) (λe, absurd e h₁) (λr, r)
|
||
|
||
lemma ne_of_not_mem_cons {a b : α} {l : list α} : a ∉ b::l → a ≠ b :=
|
||
assume nin aeqb, absurd (or.inl aeqb) nin
|
||
|
||
lemma not_mem_of_not_mem_cons {a b : α} {l : list α} : a ∉ b::l → a ∉ l :=
|
||
assume nin nainl, absurd (or.inr nainl) nin
|
||
|
||
lemma not_mem_cons_of_ne_of_not_mem {a y : α} {l : list α} : a ≠ y → a ∉ l → a ∉ y::l :=
|
||
assume p1 p2, not.intro (assume Pain, absurd (eq_or_mem_of_mem_cons Pain) (not_or p1 p2))
|
||
|
||
lemma ne_and_not_mem_of_not_mem_cons {a y : α} {l : list α} : a ∉ y::l → a ≠ y ∧ a ∉ l :=
|
||
assume p, and.intro (ne_of_not_mem_cons p) (not_mem_of_not_mem_cons p)
|
||
|
||
/- sublists -/
|
||
|
||
def sublist (l₁ l₂ : list α) := ∀ ⦃a : α⦄, a ∈ l₁ → a ∈ l₂
|
||
|
||
instance : has_subset (list α) := ⟨sublist⟩
|
||
|
||
@[simp] lemma nil_subset (l : list α) : [] ⊆ l :=
|
||
λ b i, false.elim (iff.mp (mem_nil_iff b) i)
|
||
|
||
@[simp] lemma subset.refl (l : list α) : l ⊆ l :=
|
||
λ b i, i
|
||
|
||
lemma subset.trans {l₁ l₂ l₃ : list α} (h₁ : l₁ ⊆ l₂) (h₂ : l₂ ⊆ l₃) : l₁ ⊆ l₃ :=
|
||
λ b i, h₂ (h₁ i)
|
||
|
||
@[simp] lemma subset_cons (a : α) (l : list α) : l ⊆ a::l :=
|
||
λ b i, or.inr i
|
||
|
||
lemma subset_of_cons_subset {a : α} {l₁ l₂ : list α} : a::l₁ ⊆ l₂ → l₁ ⊆ l₂ :=
|
||
λ s b i, s (mem_cons_of_mem _ i)
|
||
|
||
lemma cons_subset_cons {l₁ l₂ : list α} (a : α) (s : l₁ ⊆ l₂) : (a::l₁) ⊆ (a::l₂) :=
|
||
λ b hin, or.elim (eq_or_mem_of_mem_cons hin)
|
||
(λ e : b = a, or.inl e)
|
||
(λ i : b ∈ l₁, or.inr (s i))
|
||
|
||
@[simp] lemma subset_append_left (l₁ l₂ : list α) : l₁ ⊆ l₁++l₂ :=
|
||
λ b i, iff.mpr (mem_append_iff b l₁ l₂) (or.inl i)
|
||
|
||
@[simp] lemma subset_append_right (l₁ l₂ : list α) : l₂ ⊆ l₁++l₂ :=
|
||
λ b i, iff.mpr (mem_append_iff b l₁ l₂) (or.inr i)
|
||
|
||
lemma subset_cons_of_subset (a : α) {l₁ l₂ : list α} : l₁ ⊆ l₂ → l₁ ⊆ (a::l₂) :=
|
||
λ (s : l₁ ⊆ l₂) (a : α) (i : a ∈ l₁), or.inr (s i)
|
||
|
||
lemma subset_app_of_subset_left (l l₁ l₂ : list α) : l ⊆ l₁ → l ⊆ l₁++l₂ :=
|
||
λ (s : l ⊆ l₁) (a : α) (ainl : a ∈ l),
|
||
have a ∈ l₁, from s ainl,
|
||
mem_append_of_mem_or_mem (or.inl this)
|
||
|
||
lemma subset_app_of_subset_right (l l₁ l₂ : list α) : l ⊆ l₂ → l ⊆ l₁++l₂ :=
|
||
λ (s : l ⊆ l₂) (a : α) (ainl : a ∈ l),
|
||
have a ∈ l₂, from s ainl,
|
||
mem_append_of_mem_or_mem (or.inr this)
|
||
|
||
lemma cons_subset_of_subset_of_mem {a : α} {l m : list α} (ainm : a ∈ m) (lsubm : l ⊆ m) :
|
||
a::l ⊆ m :=
|
||
take b, suppose b ∈ a::l,
|
||
or.elim (eq_or_mem_of_mem_cons this)
|
||
(suppose b = a, begin subst b, exact ainm end)
|
||
(suppose b ∈ l, lsubm this)
|
||
|
||
lemma app_subset_of_subset_of_subset {l₁ l₂ l : list α} (l₁subl : l₁ ⊆ l) (l₂subl : l₂ ⊆ l) :
|
||
l₁ ++ l₂ ⊆ l :=
|
||
take a, suppose a ∈ l₁ ++ l₂,
|
||
or.elim (mem_or_mem_of_mem_append this)
|
||
(suppose a ∈ l₁, l₁subl this)
|
||
(suppose a ∈ l₂, l₂subl this)
|
||
|
||
lemma eq_nil_of_map_eq_nil {f : α → β} {l :list α} (h : map f l = nil) : l = nil :=
|
||
eq_nil_of_length_eq_zero (begin rw -(length_map f l), simp [h] end)
|
||
|
||
lemma mem_map (f : α → β) {a : α} {l : list α} (h : a ∈ l) : f a ∈ map f l :=
|
||
begin
|
||
induction l with b l' ih,
|
||
{simp at h, contradiction },
|
||
{simp, simp at h, cases h with h h,
|
||
{simph},
|
||
{exact or.inr (ih h)}}
|
||
end
|
||
|
||
lemma exists_of_mem_map {f : α → β} {b : β} {l : list α} (h : b ∈ map f l) : ∃ a, a ∈ l ∧ f a = b :=
|
||
begin
|
||
induction l with c l' ih,
|
||
{simp at h, contradiction},
|
||
{cases (eq_or_mem_of_mem_cons h) with h h,
|
||
{existsi c, simp [h]},
|
||
{cases ih h with a ha, cases ha with ha₁ ha₂,
|
||
existsi a, simph }}
|
||
end
|
||
|
||
lemma eq_of_map_const {b₁ b₂ : β} : ∀ {l : list α}, b₁ ∈ map (function.const α b₂) l → b₁ = b₂
|
||
| [] h := absurd h (not_mem_nil b₁)
|
||
| (a::l) h :=
|
||
or.elim (eq_or_mem_of_mem_cons h)
|
||
(suppose b₁ = b₂, this)
|
||
(suppose b₁ ∈ map (function.const α b₂) l, eq_of_map_const this)
|
||
|
||
end list
|