2015-04-10 05:19:52 -07:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2015 Leonardo de Moura. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
2015-11-21 15:12:45 -05:00
|
|
|
|
Authors: Leonardo de Moura, Haitao Zhang, Floris van Doorn
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
2015-05-10 17:44:08 +10:00
|
|
|
|
List combinators.
|
2015-04-10 05:19:52 -07:00
|
|
|
|
-/
|
2015-07-06 12:17:57 -07:00
|
|
|
|
import data.list.basic data.equiv
|
2015-10-14 12:27:09 -07:00
|
|
|
|
open nat prod decidable function helper_tactics algebra
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
|
|
|
|
namespace list
|
|
|
|
|
variables {A B C : Type}
|
2015-11-29 23:08:43 -08:00
|
|
|
|
|
|
|
|
|
section replicate
|
|
|
|
|
|
|
|
|
|
-- 'replicate i n' returns the list contain i copies of n.
|
|
|
|
|
definition replicate : ℕ → A → list A
|
|
|
|
|
| 0 a := []
|
|
|
|
|
| (succ n) a := a :: replicate n a
|
|
|
|
|
|
|
|
|
|
theorem length_replicate : ∀ (i : ℕ) (a : A), length (replicate i a) = i
|
|
|
|
|
| 0 a := rfl
|
|
|
|
|
| (succ i) a := calc
|
|
|
|
|
length (replicate (succ i) a) = length (replicate i a) + 1 : rfl
|
|
|
|
|
... = i + 1 : length_replicate
|
|
|
|
|
end replicate
|
|
|
|
|
|
2015-04-11 18:22:22 -07:00
|
|
|
|
/- map -/
|
2015-04-10 05:19:52 -07:00
|
|
|
|
definition map (f : A → B) : list A → list B
|
|
|
|
|
| [] := []
|
|
|
|
|
| (a :: l) := f a :: map l
|
|
|
|
|
|
|
|
|
|
theorem map_nil (f : A → B) : map f [] = []
|
|
|
|
|
|
|
|
|
|
theorem map_cons (f : A → B) (a : A) (l : list A) : map f (a :: l) = f a :: map f l
|
|
|
|
|
|
2015-11-21 02:05:01 -05:00
|
|
|
|
lemma map_concat (f : A → B) (a : A) : Πl, map f (concat a l) = concat (f a) (map f l)
|
|
|
|
|
| nil := rfl
|
|
|
|
|
| (b::l) := begin rewrite [concat_cons, +map_cons, concat_cons, map_concat] end
|
|
|
|
|
|
2015-07-13 14:38:31 -07:00
|
|
|
|
lemma map_append (f : A → B) : ∀ l₁ l₂, map f (l₁++l₂) = (map f l₁)++(map f l₂)
|
|
|
|
|
| nil := take l, rfl
|
|
|
|
|
| (a::l) := take l', begin rewrite [append_cons, *map_cons, append_cons, map_append] end
|
|
|
|
|
|
2015-11-21 02:05:01 -05:00
|
|
|
|
lemma map_reverse (f : A → B) : Πl, map f (reverse l) = reverse (map f l)
|
|
|
|
|
| nil := rfl
|
|
|
|
|
| (b::l) := begin rewrite [reverse_cons, +map_cons, reverse_cons, map_concat, map_reverse] end
|
|
|
|
|
|
2015-07-13 14:38:31 -07:00
|
|
|
|
lemma map_singleton (f : A → B) (a : A) : map f [a] = [f a] := rfl
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem map_id [simp] : ∀ l : list A, map id l = l
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| [] := rfl
|
|
|
|
|
| (x::xs) := begin rewrite [map_cons, map_id] end
|
|
|
|
|
|
2015-06-04 18:51:34 +10:00
|
|
|
|
theorem map_id' {f : A → A} (H : ∀x, f x = x) : ∀ l : list A, map f l = l
|
|
|
|
|
| [] := rfl
|
|
|
|
|
| (x::xs) := begin rewrite [map_cons, H, map_id'] end
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem map_map [simp] (g : B → C) (f : A → B) : ∀ l, map g (map f l) = map (g ∘ f) l
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| [] := rfl
|
|
|
|
|
| (a :: l) :=
|
|
|
|
|
show (g ∘ f) a :: map g (map f l) = map (g ∘ f) (a :: l),
|
|
|
|
|
by rewrite (map_map l)
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem length_map [simp] (f : A → B) : ∀ l : list A, length (map f l) = length l
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| [] := by esimp
|
|
|
|
|
| (a :: l) :=
|
|
|
|
|
show length (map f l) + 1 = length l + 1,
|
2015-06-04 15:09:52 -07:00
|
|
|
|
by rewrite (length_map l)
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
|
|
|
|
theorem mem_map {A B : Type} (f : A → B) : ∀ {a l}, a ∈ l → f a ∈ map f l
|
|
|
|
|
| a [] i := absurd i !not_mem_nil
|
|
|
|
|
| a (x::xs) i := or.elim (eq_or_mem_of_mem_cons i)
|
2015-07-20 22:00:05 -07:00
|
|
|
|
(suppose a = x, by rewrite [this, map_cons]; apply mem_cons)
|
|
|
|
|
(suppose a ∈ xs, or.inr (mem_map this))
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
2015-05-10 17:44:08 +10:00
|
|
|
|
theorem exists_of_mem_map {A B : Type} {f : A → B} {b : B} :
|
|
|
|
|
∀{l}, b ∈ map f l → ∃a, a ∈ l ∧ f a = b
|
|
|
|
|
| [] H := false.elim H
|
|
|
|
|
| (c::l) H := or.elim (iff.mp !mem_cons_iff H)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose b = f c,
|
|
|
|
|
exists.intro c (and.intro !mem_cons (eq.symm this)))
|
|
|
|
|
(suppose b ∈ map f l,
|
|
|
|
|
obtain a (Hl : a ∈ l) (Hr : f a = b), from exists_of_mem_map this,
|
2015-05-11 09:14:48 -07:00
|
|
|
|
exists.intro a (and.intro (mem_cons_of_mem _ Hl) Hr))
|
2015-05-10 17:44:08 +10:00
|
|
|
|
|
2015-04-11 13:52:50 -07:00
|
|
|
|
theorem eq_of_map_const {A B : Type} {b₁ b₂ : B} : ∀ {l : list A}, b₁ ∈ map (const A b₂) l → b₁ = b₂
|
|
|
|
|
| [] h := absurd h !not_mem_nil
|
|
|
|
|
| (a::l) h :=
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons h)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose b₁ = b₂, this)
|
|
|
|
|
(suppose b₁ ∈ map (const A b₂) l, eq_of_map_const this)
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-04-10 05:19:52 -07:00
|
|
|
|
definition map₂ (f : A → B → C) : list A → list B → list C
|
|
|
|
|
| [] _ := []
|
|
|
|
|
| _ [] := []
|
|
|
|
|
| (x::xs) (y::ys) := f x y :: map₂ xs ys
|
|
|
|
|
|
2015-11-29 23:08:43 -08:00
|
|
|
|
theorem map₂_nil1 (f : A → B → C) : ∀ (l : list B), map₂ f [] l = []
|
|
|
|
|
| [] := rfl
|
|
|
|
|
| (a::y) := rfl
|
|
|
|
|
|
|
|
|
|
theorem map₂_nil2 (f : A → B → C) : ∀ (l : list A), map₂ f l [] = []
|
|
|
|
|
| [] := rfl
|
|
|
|
|
| (a::y) := rfl
|
|
|
|
|
|
|
|
|
|
theorem length_map₂ : ∀(f : A → B → C) x y, length (map₂ f x y) = min (length x) (length y)
|
|
|
|
|
| f [] [] := rfl
|
|
|
|
|
| f (xh::xr) [] := rfl
|
|
|
|
|
| f [] (yh::yr) := rfl
|
|
|
|
|
| f (xh::xr) (yh::yr) := calc
|
|
|
|
|
length (map₂ f (xh::xr) (yh::yr))
|
|
|
|
|
= length (map₂ f xr yr) + 1 : rfl
|
|
|
|
|
... = min (length xr) (length yr) + 1 : length_map₂
|
|
|
|
|
... = min (length (xh::xr)) (length (yh::yr)) : min_succ_succ
|
|
|
|
|
|
2015-04-11 18:22:22 -07:00
|
|
|
|
/- filter -/
|
|
|
|
|
definition filter (p : A → Prop) [h : decidable_pred p] : list A → list A
|
|
|
|
|
| [] := []
|
|
|
|
|
| (a::l) := if p a then a :: filter l else filter l
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem filter_nil [simp] (p : A → Prop) [h : decidable_pred p] : filter p [] = []
|
2015-04-11 18:22:22 -07:00
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem filter_cons_of_pos [simp] {p : A → Prop} [h : decidable_pred p] {a : A} : ∀ l, p a → filter p (a::l) = a :: filter p l :=
|
2015-04-11 18:22:22 -07:00
|
|
|
|
λ l pa, if_pos pa
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem filter_cons_of_neg [simp] {p : A → Prop} [h : decidable_pred p] {a : A} : ∀ l, ¬ p a → filter p (a::l) = filter p l :=
|
2015-04-11 18:22:22 -07:00
|
|
|
|
λ l pa, if_neg pa
|
|
|
|
|
|
|
|
|
|
theorem of_mem_filter {p : A → Prop} [h : decidable_pred p] {a : A} : ∀ {l}, a ∈ filter p l → p a
|
|
|
|
|
| [] ain := absurd ain !not_mem_nil
|
|
|
|
|
| (b::l) ain := by_cases
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(assume pb : p b,
|
2015-07-20 22:00:05 -07:00
|
|
|
|
have a ∈ b :: filter p l, by rewrite [filter_cons_of_pos _ pb at ain]; exact ain,
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons this)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose a = b, by rewrite [-this at pb]; exact pb)
|
|
|
|
|
(suppose a ∈ filter p l, of_mem_filter this))
|
|
|
|
|
(suppose ¬ p b, by rewrite [filter_cons_of_neg _ this at ain]; exact (of_mem_filter ain))
|
2015-04-11 18:22:22 -07:00
|
|
|
|
|
|
|
|
|
theorem mem_of_mem_filter {p : A → Prop} [h : decidable_pred p] {a : A} : ∀ {l}, a ∈ filter p l → a ∈ l
|
|
|
|
|
| [] ain := absurd ain !not_mem_nil
|
|
|
|
|
| (b::l) ain := by_cases
|
|
|
|
|
(λ pb : p b,
|
2015-07-20 19:44:29 -07:00
|
|
|
|
have a ∈ b :: filter p l, by rewrite [filter_cons_of_pos _ pb at ain]; exact ain,
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons this)
|
|
|
|
|
(suppose a = b, by rewrite this; exact !mem_cons)
|
|
|
|
|
(suppose a ∈ filter p l, mem_cons_of_mem _ (mem_of_mem_filter this)))
|
|
|
|
|
(suppose ¬ p b, by rewrite [filter_cons_of_neg _ this at ain]; exact (mem_cons_of_mem _ (mem_of_mem_filter ain)))
|
2015-04-11 18:22:22 -07:00
|
|
|
|
|
|
|
|
|
theorem mem_filter_of_mem {p : A → Prop} [h : decidable_pred p] {a : A} : ∀ {l}, a ∈ l → p a → a ∈ filter p l
|
|
|
|
|
| [] ain pa := absurd ain !not_mem_nil
|
|
|
|
|
| (b::l) ain pa := by_cases
|
|
|
|
|
(λ pb : p b, or.elim (eq_or_mem_of_mem_cons ain)
|
|
|
|
|
(λ aeqb : a = b, by rewrite [filter_cons_of_pos _ pb, aeqb]; exact !mem_cons)
|
|
|
|
|
(λ ainl : a ∈ l, by rewrite [filter_cons_of_pos _ pb]; exact (mem_cons_of_mem _ (mem_filter_of_mem ainl pa))))
|
|
|
|
|
(λ npb : ¬ p b, or.elim (eq_or_mem_of_mem_cons ain)
|
|
|
|
|
(λ aeqb : a = b, absurd (eq.rec_on aeqb pa) npb)
|
|
|
|
|
(λ ainl : a ∈ l, by rewrite [filter_cons_of_neg _ npb]; exact (mem_filter_of_mem ainl pa)))
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem filter_sub [simp] {p : A → Prop} [h : decidable_pred p] (l : list A) : filter p l ⊆ l :=
|
2015-04-11 18:22:22 -07:00
|
|
|
|
λ a ain, mem_of_mem_filter ain
|
|
|
|
|
|
|
|
|
|
theorem filter_append {p : A → Prop} [h : decidable_pred p] : ∀ (l₁ l₂ : list A), filter p (l₁++l₂) = filter p l₁ ++ filter p l₂
|
|
|
|
|
| [] l₂ := rfl
|
|
|
|
|
| (a::l₁) l₂ := by_cases
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose p a, by rewrite [append_cons, *filter_cons_of_pos _ this, filter_append])
|
|
|
|
|
(suppose ¬ p a, by rewrite [append_cons, *filter_cons_of_neg _ this, filter_append])
|
2015-04-11 18:22:22 -07:00
|
|
|
|
|
|
|
|
|
/- foldl & foldr -/
|
2015-04-10 05:19:52 -07:00
|
|
|
|
definition foldl (f : A → B → A) : A → list B → A
|
|
|
|
|
| a [] := a
|
|
|
|
|
| a (b :: l) := foldl (f a b) l
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldl_nil [simp] (f : A → B → A) (a : A) : foldl f a [] = a
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldl_cons [simp] (f : A → B → A) (a : A) (b : B) (l : list B) : foldl f a (b::l) = foldl f (f a b) l
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
|
|
|
|
definition foldr (f : A → B → B) : B → list A → B
|
|
|
|
|
| b [] := b
|
|
|
|
|
| b (a :: l) := f a (foldr b l)
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldr_nil [simp] (f : A → B → B) (b : B) : foldr f b [] = b
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldr_cons [simp] (f : A → B → B) (b : B) (a : A) (l : list A) : foldr f b (a::l) = f a (foldr f b l)
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
|
|
|
|
section foldl_eq_foldr
|
|
|
|
|
-- foldl and foldr coincide when f is commutative and associative
|
|
|
|
|
parameters {α : Type} {f : α → α → α}
|
|
|
|
|
hypothesis (Hcomm : ∀ a b, f a b = f b a)
|
|
|
|
|
hypothesis (Hassoc : ∀ a b c, f (f a b) c = f a (f b c))
|
|
|
|
|
include Hcomm Hassoc
|
|
|
|
|
|
|
|
|
|
theorem foldl_eq_of_comm_of_assoc : ∀ a b l, foldl f a (b::l) = f b (foldl f a l)
|
|
|
|
|
| a b nil := Hcomm a b
|
|
|
|
|
| a b (c::l) :=
|
|
|
|
|
begin
|
2015-04-29 14:39:59 -07:00
|
|
|
|
change foldl f (f (f a b) c) l = f b (foldl f (f a c) l),
|
2015-04-10 05:19:52 -07:00
|
|
|
|
rewrite -foldl_eq_of_comm_of_assoc,
|
2015-04-29 14:39:59 -07:00
|
|
|
|
change foldl f (f (f a b) c) l = foldl f (f (f a c) b) l,
|
2015-04-10 05:19:52 -07:00
|
|
|
|
have H₁ : f (f a b) c = f (f a c) b, by rewrite [Hassoc, Hassoc, Hcomm b c],
|
|
|
|
|
rewrite H₁
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem foldl_eq_foldr : ∀ a l, foldl f a l = foldr f a l
|
|
|
|
|
| a nil := rfl
|
|
|
|
|
| a (b :: l) :=
|
|
|
|
|
begin
|
|
|
|
|
rewrite foldl_eq_of_comm_of_assoc,
|
|
|
|
|
esimp,
|
2015-04-29 14:39:59 -07:00
|
|
|
|
change f b (foldl f a l) = f b (foldr f a l),
|
2015-04-10 05:19:52 -07:00
|
|
|
|
rewrite foldl_eq_foldr
|
|
|
|
|
end
|
|
|
|
|
end foldl_eq_foldr
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldl_append [simp] (f : B → A → B) : ∀ (b : B) (l₁ l₂ : list A), foldl f b (l₁++l₂) = foldl f (foldl f b l₁) l₂
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| b [] l₂ := rfl
|
|
|
|
|
| b (a::l₁) l₂ := by rewrite [append_cons, *foldl_cons, foldl_append]
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem foldr_append [simp] (f : A → B → B) : ∀ (b : B) (l₁ l₂ : list A), foldr f b (l₁++l₂) = foldr f (foldr f b l₂) l₁
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| b [] l₂ := rfl
|
|
|
|
|
| b (a::l₁) l₂ := by rewrite [append_cons, *foldr_cons, foldr_append]
|
|
|
|
|
|
2015-04-11 18:22:22 -07:00
|
|
|
|
/- all & any -/
|
2015-04-10 15:32:44 -07:00
|
|
|
|
definition all (l : list A) (p : A → Prop) : Prop :=
|
2015-04-10 05:19:52 -07:00
|
|
|
|
foldr (λ a r, p a ∧ r) true l
|
|
|
|
|
|
2015-04-10 15:32:44 -07:00
|
|
|
|
definition any (l : list A) (p : A → Prop) : Prop :=
|
2015-04-10 05:19:52 -07:00
|
|
|
|
foldr (λ a r, p a ∨ r) false l
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem all_nil_eq [simp] (p : A → Prop) : all [] p = true
|
2015-04-10 07:01:14 -07:00
|
|
|
|
|
2015-05-10 07:33:56 +10:00
|
|
|
|
theorem all_nil (p : A → Prop) : all [] p := trivial
|
|
|
|
|
|
|
|
|
|
theorem all_cons_eq (p : A → Prop) (a : A) (l : list A) : all (a::l) p = (p a ∧ all l p)
|
|
|
|
|
|
|
|
|
|
theorem all_cons {p : A → Prop} {a : A} {l : list A} (H1 : p a) (H2 : all l p) : all (a::l) p :=
|
|
|
|
|
and.intro H1 H2
|
2015-04-10 07:01:14 -07:00
|
|
|
|
|
2015-04-10 15:32:44 -07:00
|
|
|
|
theorem all_of_all_cons {p : A → Prop} {a : A} {l : list A} : all (a::l) p → all l p :=
|
2015-05-10 07:33:56 +10:00
|
|
|
|
assume h, by rewrite [all_cons_eq at h]; exact (and.elim_right h)
|
2015-04-10 15:32:44 -07:00
|
|
|
|
|
|
|
|
|
theorem of_all_cons {p : A → Prop} {a : A} {l : list A} : all (a::l) p → p a :=
|
2015-05-10 07:33:56 +10:00
|
|
|
|
assume h, by rewrite [all_cons_eq at h]; exact (and.elim_left h)
|
2015-04-10 15:32:44 -07:00
|
|
|
|
|
|
|
|
|
theorem all_cons_of_all {p : A → Prop} {a : A} {l : list A} : p a → all l p → all (a::l) p :=
|
|
|
|
|
assume pa alllp, and.intro pa alllp
|
|
|
|
|
|
|
|
|
|
theorem all_implies {p q : A → Prop} : ∀ {l}, all l p → (∀ x, p x → q x) → all l q
|
|
|
|
|
| [] h₁ h₂ := trivial
|
|
|
|
|
| (a::l) h₁ h₂ :=
|
2015-07-20 22:00:05 -07:00
|
|
|
|
have all l q, from all_implies (all_of_all_cons h₁) h₂,
|
2015-07-20 19:44:29 -07:00
|
|
|
|
have q a, from h₂ a (of_all_cons h₁),
|
2015-07-20 22:00:05 -07:00
|
|
|
|
all_cons_of_all this `all l q`
|
2015-04-10 15:32:44 -07:00
|
|
|
|
|
|
|
|
|
theorem of_mem_of_all {p : A → Prop} {a : A} : ∀ {l}, a ∈ l → all l p → p a
|
2015-04-10 07:01:14 -07:00
|
|
|
|
| [] h₁ h₂ := absurd h₁ !not_mem_nil
|
|
|
|
|
| (b::l) h₁ h₂ :=
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons h₁)
|
2015-07-20 22:00:05 -07:00
|
|
|
|
(suppose a = b,
|
|
|
|
|
by rewrite [all_cons_eq at h₂, -this at h₂]; exact (and.elim_left h₂))
|
|
|
|
|
(suppose a ∈ l,
|
2015-07-20 19:44:29 -07:00
|
|
|
|
have all l p, by rewrite [all_cons_eq at h₂]; exact (and.elim_right h₂),
|
2015-07-20 22:00:05 -07:00
|
|
|
|
of_mem_of_all `a ∈ l` this)
|
2015-04-10 07:01:14 -07:00
|
|
|
|
|
2015-05-10 07:33:56 +10:00
|
|
|
|
theorem all_of_forall {p : A → Prop} : ∀ {l}, (∀a, a ∈ l → p a) → all l p
|
|
|
|
|
| [] H := !all_nil
|
|
|
|
|
| (a::l) H := all_cons (H a !mem_cons)
|
|
|
|
|
(all_of_forall (λ a' H', H a' (mem_cons_of_mem _ H')))
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem any_nil [simp] (p : A → Prop) : any [] p = false
|
2015-04-10 15:32:44 -07:00
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem any_cons [simp] (p : A → Prop) (a : A) (l : list A) : any (a::l) p = (p a ∨ any l p)
|
2015-04-10 07:01:14 -07:00
|
|
|
|
|
2015-05-10 07:33:56 +10:00
|
|
|
|
theorem any_of_mem {p : A → Prop} {a : A} : ∀ {l}, a ∈ l → p a → any l p
|
2015-04-10 15:32:44 -07:00
|
|
|
|
| [] i h := absurd i !not_mem_nil
|
|
|
|
|
| (b::l) i h :=
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons i)
|
2015-07-20 22:00:05 -07:00
|
|
|
|
(suppose a = b, by rewrite [-this]; exact (or.inl h))
|
|
|
|
|
(suppose a ∈ l,
|
|
|
|
|
have any l p, from any_of_mem this h,
|
|
|
|
|
or.inr this)
|
2015-04-10 07:01:14 -07:00
|
|
|
|
|
2015-05-10 07:33:56 +10:00
|
|
|
|
theorem exists_of_any {p : A → Prop} : ∀{l : list A}, any l p → ∃a, a ∈ l ∧ p a
|
|
|
|
|
| [] H := false.elim H
|
|
|
|
|
| (b::l) H := or.elim H
|
|
|
|
|
(assume H1 : p b, exists.intro b (and.intro !mem_cons H1))
|
|
|
|
|
(assume H1 : any l p,
|
2015-05-11 09:14:48 -07:00
|
|
|
|
obtain a (H2l : a ∈ l) (H2r : p a), from exists_of_any H1,
|
|
|
|
|
exists.intro a (and.intro (mem_cons_of_mem b H2l) H2r))
|
2015-05-10 07:33:56 +10:00
|
|
|
|
|
2015-04-10 15:32:44 -07:00
|
|
|
|
definition decidable_all (p : A → Prop) [H : decidable_pred p] : ∀ l, decidable (all l p)
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| [] := decidable_true
|
|
|
|
|
| (a :: l) :=
|
|
|
|
|
match H a with
|
|
|
|
|
| inl Hp₁ :=
|
|
|
|
|
match decidable_all l with
|
|
|
|
|
| inl Hp₂ := inl (and.intro Hp₁ Hp₂)
|
|
|
|
|
| inr Hn₂ := inr (not_and_of_not_right (p a) Hn₂)
|
|
|
|
|
end
|
2015-04-10 15:32:44 -07:00
|
|
|
|
| inr Hn := inr (not_and_of_not_left (all l p) Hn)
|
2015-04-10 05:19:52 -07:00
|
|
|
|
end
|
|
|
|
|
|
2015-04-10 15:32:44 -07:00
|
|
|
|
definition decidable_any (p : A → Prop) [H : decidable_pred p] : ∀ l, decidable (any l p)
|
2015-04-10 05:19:52 -07:00
|
|
|
|
| [] := decidable_false
|
|
|
|
|
| (a :: l) :=
|
|
|
|
|
match H a with
|
|
|
|
|
| inl Hp := inl (or.inl Hp)
|
|
|
|
|
| inr Hn₁ :=
|
|
|
|
|
match decidable_any l with
|
|
|
|
|
| inl Hp₂ := inl (or.inr Hp₂)
|
|
|
|
|
| inr Hn₂ := inr (not_or Hn₁ Hn₂)
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2015-04-11 18:22:22 -07:00
|
|
|
|
/- zip & unzip -/
|
2015-04-10 05:19:52 -07:00
|
|
|
|
definition zip (l₁ : list A) (l₂ : list B) : list (A × B) :=
|
|
|
|
|
map₂ (λ a b, (a, b)) l₁ l₂
|
|
|
|
|
|
|
|
|
|
definition unzip : list (A × B) → list A × list B
|
|
|
|
|
| [] := ([], [])
|
|
|
|
|
| ((a, b) :: l) :=
|
|
|
|
|
match unzip l with
|
|
|
|
|
| (la, lb) := (a :: la, b :: lb)
|
|
|
|
|
end
|
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem unzip_nil [simp] : unzip (@nil (A × B)) = ([], [])
|
2015-04-10 05:19:52 -07:00
|
|
|
|
|
2015-07-22 17:21:47 -07:00
|
|
|
|
theorem unzip_cons [simp] (a : A) (b : B) (l : list (A × B)) :
|
2015-04-10 05:19:52 -07:00
|
|
|
|
unzip ((a, b) :: l) = match unzip l with (la, lb) := (a :: la, b :: lb) end :=
|
|
|
|
|
rfl
|
|
|
|
|
|
|
|
|
|
theorem zip_unzip : ∀ (l : list (A × B)), zip (pr₁ (unzip l)) (pr₂ (unzip l)) = l
|
|
|
|
|
| [] := rfl
|
|
|
|
|
| ((a, b) :: l) :=
|
|
|
|
|
begin
|
|
|
|
|
rewrite unzip_cons,
|
|
|
|
|
have r : zip (pr₁ (unzip l)) (pr₂ (unzip l)) = l, from zip_unzip l,
|
|
|
|
|
revert r,
|
2015-05-01 15:07:28 -07:00
|
|
|
|
eapply prod.cases_on (unzip l),
|
2015-04-30 11:00:39 -07:00
|
|
|
|
intro la lb r,
|
2015-04-10 05:19:52 -07:00
|
|
|
|
rewrite -r
|
|
|
|
|
end
|
|
|
|
|
|
2015-11-29 23:08:43 -08:00
|
|
|
|
section mapAccumR
|
|
|
|
|
variable {S : Type}
|
|
|
|
|
|
|
|
|
|
-- This runs a function over a list returning the intermediate results and a
|
|
|
|
|
-- a final result.
|
|
|
|
|
definition mapAccumR : (A → S → S × B) → list A → S → (S × list B)
|
|
|
|
|
| f [] c := (c, [])
|
|
|
|
|
| f (y::yr) c :=
|
|
|
|
|
let r := mapAccumR f yr c in
|
|
|
|
|
let z := f y (pr₁ r) in
|
|
|
|
|
(pr₁ z, pr₂ z :: pr₂ r)
|
|
|
|
|
|
|
|
|
|
theorem length_mapAccumR
|
|
|
|
|
: ∀ (f : A → S → S × B) (x : list A) (s : S),
|
|
|
|
|
length (pr₂ (mapAccumR f x s)) = length x
|
|
|
|
|
| f (a::x) s := calc
|
|
|
|
|
length (pr₂ (mapAccumR f (a::x) s))
|
|
|
|
|
= length x + 1 : { length_mapAccumR f x s }
|
|
|
|
|
... = length (a::x) : rfl
|
|
|
|
|
| f [] s := calc
|
|
|
|
|
length (pr₂ (mapAccumR f [] s))
|
|
|
|
|
= 0 : rfl
|
|
|
|
|
end mapAccumR
|
|
|
|
|
|
|
|
|
|
section mapAccumR₂
|
|
|
|
|
variable {S : Type}
|
|
|
|
|
-- This runs a function over two lists returning the intermediate results and a
|
|
|
|
|
-- a final result.
|
|
|
|
|
definition mapAccumR₂
|
|
|
|
|
: (A → B → S → S × C) → list A → list B → S → S × list C
|
|
|
|
|
| f [] _ c := (c,[])
|
|
|
|
|
| f _ [] c := (c,[])
|
|
|
|
|
| f (x::xr) (y::yr) c :=
|
|
|
|
|
let r := mapAccumR₂ f xr yr c in
|
|
|
|
|
let q := f x y (pr₁ r) in
|
|
|
|
|
(pr₁ q, pr₂ q :: (pr₂ r))
|
|
|
|
|
|
|
|
|
|
theorem length_mapAccumR₂
|
|
|
|
|
: ∀ (f : A → B → S → S × C) (x : list A) (y : list B) (c : S),
|
|
|
|
|
length (pr₂ (mapAccumR₂ f x y c)) = min (length x) (length y)
|
|
|
|
|
| f (a::x) (b::y) c := calc
|
|
|
|
|
length (pr₂ (mapAccumR₂ f (a::x) (b::y) c))
|
|
|
|
|
= length (pr₂ (mapAccumR₂ f x y c)) + 1 : rfl
|
|
|
|
|
... = min (length x) (length y) + 1 : length_mapAccumR₂ f x y c
|
|
|
|
|
... = min (length (a::x)) (length (b::y)) : min_succ_succ
|
|
|
|
|
| f (a::x) [] c := rfl
|
|
|
|
|
| f [] (b::y) c := rfl
|
|
|
|
|
| f [] [] c := rfl
|
|
|
|
|
|
|
|
|
|
end mapAccumR₂
|
|
|
|
|
|
2015-04-10 05:19:52 -07:00
|
|
|
|
/- flat -/
|
|
|
|
|
definition flat (l : list (list A)) : list A :=
|
|
|
|
|
foldl append nil l
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 14:00:55 +10:00
|
|
|
|
/- product -/
|
2015-05-08 13:46:14 +10:00
|
|
|
|
section product
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
definition product : list A → list B → list (A × B)
|
2015-04-11 13:52:50 -07:00
|
|
|
|
| [] l₂ := []
|
2015-05-08 13:46:14 +10:00
|
|
|
|
| (a::l₁) l₂ := map (λ b, (a, b)) l₂ ++ product l₁ l₂
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem nil_product (l : list B) : product (@nil A) l = []
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem product_cons (a : A) (l₁ : list A) (l₂ : list B)
|
|
|
|
|
: product (a::l₁) l₂ = map (λ b, (a, b)) l₂ ++ product l₁ l₂
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem product_nil : ∀ (l : list A), product l (@nil B) = []
|
2015-04-11 13:52:50 -07:00
|
|
|
|
| [] := rfl
|
2015-05-08 13:46:14 +10:00
|
|
|
|
| (a::l) := by rewrite [product_cons, map_nil, product_nil]
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-04-11 14:30:20 -07:00
|
|
|
|
theorem eq_of_mem_map_pair₁ {a₁ a : A} {b₁ : B} {l : list B} : (a₁, b₁) ∈ map (λ b, (a, b)) l → a₁ = a :=
|
|
|
|
|
assume ain,
|
2015-07-20 22:00:05 -07:00
|
|
|
|
assert pr1 (a₁, b₁) ∈ map pr1 (map (λ b, (a, b)) l), from mem_map pr1 ain,
|
|
|
|
|
assert a₁ ∈ map (λb, a) l, by revert this; rewrite [map_map, ↑pr1]; intro this; assumption,
|
|
|
|
|
eq_of_map_const this
|
2015-04-11 14:30:20 -07:00
|
|
|
|
|
|
|
|
|
theorem mem_of_mem_map_pair₁ {a₁ a : A} {b₁ : B} {l : list B} : (a₁, b₁) ∈ map (λ b, (a, b)) l → b₁ ∈ l :=
|
|
|
|
|
assume ain,
|
2015-07-20 19:44:29 -07:00
|
|
|
|
assert pr2 (a₁, b₁) ∈ map pr2 (map (λ b, (a, b)) l), from mem_map pr2 ain,
|
|
|
|
|
assert b₁ ∈ map (λx, x) l, by rewrite [map_map at this, ↑pr2 at this]; exact this,
|
|
|
|
|
by rewrite [map_id at this]; exact this
|
2015-04-11 14:30:20 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem mem_product {a : A} {b : B} : ∀ {l₁ l₂}, a ∈ l₁ → b ∈ l₂ → (a, b) ∈ product l₁ l₂
|
2015-04-11 13:52:50 -07:00
|
|
|
|
| [] l₂ h₁ h₂ := absurd h₁ !not_mem_nil
|
|
|
|
|
| (x::l₁) l₂ h₁ h₂ :=
|
|
|
|
|
or.elim (eq_or_mem_of_mem_cons h₁)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(assume aeqx : a = x,
|
|
|
|
|
assert (a, b) ∈ map (λ b, (a, b)) l₂, from mem_map _ h₂,
|
|
|
|
|
begin rewrite [-aeqx, product_cons], exact mem_append_left _ this end)
|
|
|
|
|
(assume ainl₁ : a ∈ l₁,
|
|
|
|
|
assert (a, b) ∈ product l₁ l₂, from mem_product ainl₁ h₂,
|
|
|
|
|
begin rewrite [product_cons], exact mem_append_right _ this end)
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem mem_of_mem_product_left {a : A} {b : B} : ∀ {l₁ l₂}, (a, b) ∈ product l₁ l₂ → a ∈ l₁
|
2015-04-11 13:52:50 -07:00
|
|
|
|
| [] l₂ h := absurd h !not_mem_nil
|
|
|
|
|
| (x::l₁) l₂ h :=
|
|
|
|
|
or.elim (mem_or_mem_of_mem_append h)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose (a, b) ∈ map (λ b, (x, b)) l₂,
|
|
|
|
|
assert a = x, from eq_of_mem_map_pair₁ this,
|
|
|
|
|
by rewrite this; exact !mem_cons)
|
|
|
|
|
(suppose (a, b) ∈ product l₁ l₂,
|
|
|
|
|
have a ∈ l₁, from mem_of_mem_product_left this,
|
|
|
|
|
mem_cons_of_mem _ this)
|
2015-04-11 13:52:50 -07:00
|
|
|
|
|
2015-05-08 13:46:14 +10:00
|
|
|
|
theorem mem_of_mem_product_right {a : A} {b : B} : ∀ {l₁ l₂}, (a, b) ∈ product l₁ l₂ → b ∈ l₂
|
2015-04-11 13:52:50 -07:00
|
|
|
|
| [] l₂ h := absurd h !not_mem_nil
|
|
|
|
|
| (x::l₁) l₂ h :=
|
|
|
|
|
or.elim (mem_or_mem_of_mem_append h)
|
2015-07-20 19:44:29 -07:00
|
|
|
|
(suppose (a, b) ∈ map (λ b, (x, b)) l₂,
|
|
|
|
|
mem_of_mem_map_pair₁ this)
|
|
|
|
|
(suppose (a, b) ∈ product l₁ l₂,
|
|
|
|
|
mem_of_mem_product_right this)
|
2015-06-04 15:09:52 -07:00
|
|
|
|
|
|
|
|
|
theorem length_product : ∀ (l₁ : list A) (l₂ : list B), length (product l₁ l₂) = length l₁ * length l₂
|
|
|
|
|
| [] l₂ := by rewrite [length_nil, zero_mul]
|
|
|
|
|
| (x::l₁) l₂ :=
|
2015-07-20 19:44:29 -07:00
|
|
|
|
assert length (product l₁ l₂) = length l₁ * length l₂, from length_product l₁ l₂,
|
2015-06-04 15:09:52 -07:00
|
|
|
|
by rewrite [product_cons, length_append, length_cons,
|
2015-10-14 12:27:09 -07:00
|
|
|
|
length_map, this, right_distrib, one_mul, add.comm]
|
2015-05-08 13:46:14 +10:00
|
|
|
|
end product
|
2015-06-05 18:18:59 +10:00
|
|
|
|
|
|
|
|
|
-- new for list/comb dependent map theory
|
|
|
|
|
definition dinj₁ (p : A → Prop) (f : Π a, p a → B) := ∀ ⦃a1 a2⦄ (h1 : p a1) (h2 : p a2), a1 ≠ a2 → (f a1 h1) ≠ (f a2 h2)
|
|
|
|
|
definition dinj (p : A → Prop) (f : Π a, p a → B) := ∀ ⦃a1 a2⦄ (h1 : p a1) (h2 : p a2), (f a1 h1) = (f a2 h2) → a1 = a2
|
|
|
|
|
|
|
|
|
|
definition dmap (p : A → Prop) [h : decidable_pred p] (f : Π a, p a → B) : list A → list B
|
|
|
|
|
| [] := []
|
|
|
|
|
| (a::l) := if P : (p a) then cons (f a P) (dmap l) else (dmap l)
|
|
|
|
|
|
|
|
|
|
-- properties of dmap
|
|
|
|
|
section dmap
|
|
|
|
|
|
|
|
|
|
variable {p : A → Prop}
|
|
|
|
|
variable [h : decidable_pred p]
|
|
|
|
|
include h
|
|
|
|
|
variable {f : Π a, p a → B}
|
|
|
|
|
|
|
|
|
|
lemma dmap_nil : dmap p f [] = [] := rfl
|
|
|
|
|
lemma dmap_cons_of_pos {a : A} (P : p a) : ∀ l, dmap p f (a::l) = (f a P) :: dmap p f l :=
|
|
|
|
|
λ l, dif_pos P
|
|
|
|
|
lemma dmap_cons_of_neg {a : A} (P : ¬ p a) : ∀ l, dmap p f (a::l) = dmap p f l :=
|
|
|
|
|
λ l, dif_neg P
|
|
|
|
|
|
2015-06-10 23:16:53 -07:00
|
|
|
|
lemma mem_dmap : ∀ {l : list A} {a} (Pa : p a), a ∈ l → (f a Pa) ∈ dmap p f l
|
2015-06-05 18:18:59 +10:00
|
|
|
|
| [] := take a Pa Pinnil, by contradiction
|
|
|
|
|
| (a::l) := take b Pb Pbin, or.elim (eq_or_mem_of_mem_cons Pbin)
|
|
|
|
|
(assume Pbeqa, begin
|
|
|
|
|
rewrite [eq.symm Pbeqa, dmap_cons_of_pos Pb],
|
|
|
|
|
exact !mem_cons
|
|
|
|
|
end)
|
|
|
|
|
(assume Pbinl,
|
|
|
|
|
decidable.rec_on (h a)
|
|
|
|
|
(assume Pa, begin
|
|
|
|
|
rewrite [dmap_cons_of_pos Pa],
|
|
|
|
|
apply mem_cons_of_mem,
|
2015-06-10 23:16:53 -07:00
|
|
|
|
exact mem_dmap Pb Pbinl
|
2015-06-05 18:18:59 +10:00
|
|
|
|
end)
|
|
|
|
|
(assume nPa, begin
|
|
|
|
|
rewrite [dmap_cons_of_neg nPa],
|
2015-06-10 23:16:53 -07:00
|
|
|
|
exact mem_dmap Pb Pbinl
|
2015-06-05 18:18:59 +10:00
|
|
|
|
end))
|
|
|
|
|
|
2015-06-10 23:16:53 -07:00
|
|
|
|
lemma exists_of_mem_dmap : ∀ {l : list A} {b : B}, b ∈ dmap p f l → ∃ a P, a ∈ l ∧ b = f a P
|
|
|
|
|
| [] := take b, by rewrite dmap_nil; contradiction
|
|
|
|
|
| (a::l) := take b, decidable.rec_on (h a)
|
|
|
|
|
(assume Pa, begin
|
|
|
|
|
rewrite [dmap_cons_of_pos Pa, mem_cons_iff],
|
|
|
|
|
intro Pb, cases Pb with Peq Pin,
|
|
|
|
|
exact exists.intro a (exists.intro Pa (and.intro !mem_cons Peq)),
|
|
|
|
|
assert Pex : ∃ (a : A) (P : p a), a ∈ l ∧ b = f a P, exact exists_of_mem_dmap Pin,
|
|
|
|
|
cases Pex with a' Pex', cases Pex' with Pa' P',
|
|
|
|
|
exact exists.intro a' (exists.intro Pa' (and.intro (mem_cons_of_mem a (and.left P')) (and.right P')))
|
|
|
|
|
end)
|
|
|
|
|
(assume nPa, begin
|
|
|
|
|
rewrite [dmap_cons_of_neg nPa],
|
|
|
|
|
intro Pin,
|
|
|
|
|
assert Pex : ∃ (a : A) (P : p a), a ∈ l ∧ b = f a P, exact exists_of_mem_dmap Pin,
|
|
|
|
|
cases Pex with a' Pex', cases Pex' with Pa' P',
|
|
|
|
|
exact exists.intro a' (exists.intro Pa' (and.intro (mem_cons_of_mem a (and.left P')) (and.right P')))
|
|
|
|
|
end)
|
|
|
|
|
|
2015-06-14 16:28:51 -07:00
|
|
|
|
lemma map_dmap_of_inv_of_pos {g : B → A} (Pinv : ∀ a (Pa : p a), g (f a Pa) = a) :
|
2015-06-05 18:18:59 +10:00
|
|
|
|
∀ {l : list A}, (∀ ⦃a⦄, a ∈ l → p a) → map g (dmap p f l) = l
|
|
|
|
|
| [] := assume Pl, by rewrite [dmap_nil, map_nil]
|
|
|
|
|
| (a::l) := assume Pal,
|
|
|
|
|
assert Pa : p a, from Pal a !mem_cons,
|
|
|
|
|
assert Pl : ∀ a, a ∈ l → p a,
|
|
|
|
|
from take x Pxin, Pal x (mem_cons_of_mem a Pxin),
|
2015-06-14 16:28:51 -07:00
|
|
|
|
by rewrite [dmap_cons_of_pos Pa, map_cons, Pinv, map_dmap_of_inv_of_pos Pl]
|
2015-06-05 18:18:59 +10:00
|
|
|
|
|
2015-06-14 16:28:51 -07:00
|
|
|
|
lemma mem_of_dinj_of_mem_dmap (Pdi : dinj p f) :
|
2015-06-10 23:16:53 -07:00
|
|
|
|
∀ {l : list A} {a} (Pa : p a), (f a Pa) ∈ dmap p f l → a ∈ l
|
2015-06-05 18:18:59 +10:00
|
|
|
|
| [] := take a Pa Pinnil, by contradiction
|
|
|
|
|
| (b::l) := take a Pa Pmap,
|
|
|
|
|
decidable.rec_on (h b)
|
|
|
|
|
(λ Pb, begin
|
|
|
|
|
rewrite (dmap_cons_of_pos Pb) at Pmap,
|
|
|
|
|
rewrite mem_cons_iff at Pmap,
|
|
|
|
|
rewrite mem_cons_iff,
|
|
|
|
|
apply (or_of_or_of_imp_of_imp Pmap),
|
|
|
|
|
apply Pdi,
|
2015-06-14 16:28:51 -07:00
|
|
|
|
apply mem_of_dinj_of_mem_dmap Pa
|
2015-06-05 18:18:59 +10:00
|
|
|
|
end)
|
|
|
|
|
(λ nPb, begin
|
|
|
|
|
rewrite (dmap_cons_of_neg nPb) at Pmap,
|
|
|
|
|
apply mem_cons_of_mem,
|
2015-06-14 16:28:51 -07:00
|
|
|
|
exact mem_of_dinj_of_mem_dmap Pa Pmap
|
2015-06-05 18:18:59 +10:00
|
|
|
|
end)
|
|
|
|
|
|
2015-06-14 16:28:51 -07:00
|
|
|
|
lemma not_mem_dmap_of_dinj_of_not_mem (Pdi : dinj p f) {l : list A} {a} (Pa : p a) :
|
2015-06-05 18:18:59 +10:00
|
|
|
|
a ∉ l → (f a Pa) ∉ dmap p f l :=
|
2015-07-24 11:56:18 -04:00
|
|
|
|
not.mto (mem_of_dinj_of_mem_dmap Pdi Pa)
|
2015-06-05 18:18:59 +10:00
|
|
|
|
|
|
|
|
|
end dmap
|
|
|
|
|
|
2015-07-06 12:17:57 -07:00
|
|
|
|
section
|
|
|
|
|
open equiv
|
2015-08-15 09:48:05 -07:00
|
|
|
|
definition list_equiv_of_equiv {A B : Type} : A ≃ B → list A ≃ list B
|
2015-07-06 12:17:57 -07:00
|
|
|
|
| (mk f g l r) :=
|
|
|
|
|
mk (map f) (map g)
|
2015-08-15 09:48:05 -07:00
|
|
|
|
begin intros, rewrite [map_map, id_of_left_inverse l, map_id], try reflexivity end
|
|
|
|
|
begin intros, rewrite [map_map, id_of_righ_inverse r, map_id], try reflexivity end
|
2015-08-10 16:04:02 -07:00
|
|
|
|
|
|
|
|
|
private definition to_nat : list nat → nat
|
|
|
|
|
| [] := 0
|
|
|
|
|
| (x::xs) := succ (mkpair (to_nat xs) x)
|
|
|
|
|
|
|
|
|
|
open prod.ops
|
|
|
|
|
|
|
|
|
|
private definition of_nat.F : Π (n : nat), (Π m, m < n → list nat) → list nat
|
|
|
|
|
| 0 f := []
|
|
|
|
|
| (succ n) f := (unpair n).2 :: f (unpair n).1 (unpair_lt n)
|
|
|
|
|
|
|
|
|
|
private definition of_nat : nat → list nat :=
|
|
|
|
|
well_founded.fix of_nat.F
|
|
|
|
|
|
|
|
|
|
private lemma of_nat_zero : of_nat 0 = [] :=
|
|
|
|
|
well_founded.fix_eq of_nat.F 0
|
|
|
|
|
|
|
|
|
|
private lemma of_nat_succ (n : nat)
|
|
|
|
|
: of_nat (succ n) = (unpair n).2 :: of_nat (unpair n).1 :=
|
|
|
|
|
well_founded.fix_eq of_nat.F (succ n)
|
|
|
|
|
|
|
|
|
|
private lemma to_nat_of_nat (n : nat) : to_nat (of_nat n) = n :=
|
|
|
|
|
nat.case_strong_induction_on n
|
|
|
|
|
_
|
|
|
|
|
(λ n ih,
|
|
|
|
|
begin
|
|
|
|
|
rewrite of_nat_succ, unfold to_nat,
|
|
|
|
|
have to_nat (of_nat (unpair n).1) = (unpair n).1, from ih _ (le_of_lt_succ (unpair_lt n)),
|
|
|
|
|
rewrite this, rewrite mkpair_unpair
|
|
|
|
|
end)
|
|
|
|
|
|
|
|
|
|
private lemma of_nat_to_nat : ∀ (l : list nat), of_nat (to_nat l) = l
|
|
|
|
|
| [] := rfl
|
|
|
|
|
| (x::xs) := begin unfold to_nat, rewrite of_nat_succ, rewrite *unpair_mkpair, esimp, congruence, apply of_nat_to_nat end
|
|
|
|
|
|
2015-08-15 09:48:05 -07:00
|
|
|
|
definition list_nat_equiv_nat : list nat ≃ nat :=
|
2015-08-10 16:04:02 -07:00
|
|
|
|
mk to_nat of_nat of_nat_to_nat to_nat_of_nat
|
|
|
|
|
|
2015-08-15 09:48:05 -07:00
|
|
|
|
definition list_equiv_self_of_equiv_nat {A : Type} : A ≃ nat → list A ≃ A :=
|
2015-08-10 16:04:02 -07:00
|
|
|
|
suppose A ≃ nat, calc
|
|
|
|
|
list A ≃ list nat : list_equiv_of_equiv this
|
|
|
|
|
... ≃ nat : list_nat_equiv_nat
|
|
|
|
|
... ≃ A : this
|
2015-07-06 12:17:57 -07:00
|
|
|
|
end
|
2015-04-10 05:19:52 -07:00
|
|
|
|
end list
|
|
|
|
|
|
|
|
|
|
attribute list.decidable_any [instance]
|
|
|
|
|
attribute list.decidable_all [instance]
|