2015-01-07 01:44:04 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2014 Floris van Doorn. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Authors: Floris van Doorn, Leonardo de Moura, Jeremy Avigad
|
2014-08-22 23:36:47 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
The order relation on the natural numbers.
|
|
|
|
|
-/
|
2015-02-01 14:55:12 +00:00
|
|
|
|
import data.nat.basic algebra.ordered_ring
|
2015-10-14 19:27:09 +00:00
|
|
|
|
open eq.ops algebra
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
|
|
|
|
namespace nat
|
2015-05-12 12:49:05 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- lt and le -/
|
2015-05-12 12:49:05 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem le_of_lt_or_eq {m n : ℕ} (H : m < n ∨ m = n) : m ≤ n :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
le_of_eq_or_lt (or.swap H)
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem lt_or_eq_of_le {m n : ℕ} (H : m ≤ n) : m < n ∨ m = n :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
or.swap (eq_or_lt_of_le H)
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem le_iff_lt_or_eq (m n : ℕ) : m ≤ n ↔ m < n ∨ m = n :=
|
|
|
|
|
iff.intro nat.lt_or_eq_of_le nat.le_of_lt_or_eq
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem lt_of_le_and_ne {m n : ℕ} (H1 : m ≤ n) : m ≠ n → m < n :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
or_resolve_right (eq_or_lt_of_le H1)
|
2014-12-26 21:25:05 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem lt_iff_le_and_ne (m n : ℕ) : m < n ↔ m ≤ n ∧ m ≠ n :=
|
2015-01-07 01:44:04 +00:00
|
|
|
|
iff.intro
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(take H, and.intro (le_of_lt H) (take H1, !lt.irrefl (H1 ▸ H)))
|
2015-10-22 19:31:58 +00:00
|
|
|
|
(and.rec nat.lt_of_le_and_ne)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem le_add_right (n k : ℕ) : n ≤ n + k :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
nat.rec !le.refl (λ k, le_succ_of_le) k
|
2014-11-22 08:15:51 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem le_add_left (n m : ℕ): n ≤ m + n :=
|
|
|
|
|
!add.comm ▸ !le_add_right
|
|
|
|
|
|
|
|
|
|
theorem le.intro {n m k : ℕ} (h : n + k = m) : n ≤ m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
h ▸ !le_add_right
|
2014-11-22 08:15:51 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
theorem le.elim {n m : ℕ} : n ≤ m → ∃ k, n + k = m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
le.rec (exists.intro 0 rfl) (λm h, Exists.rec
|
|
|
|
|
(λ k H, exists.intro (succ k) (H ▸ rfl)))
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem le.total {m n : ℕ} : m ≤ n ∨ n ≤ m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
or.imp_left le_of_lt !lt_or_ge
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- addition -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem add_le_add_left {n m : ℕ} (H : n ≤ m) (k : ℕ) : k + n ≤ k + m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
obtain l Hl, from le.elim H, le.intro (Hl ▸ !add.assoc)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem add_le_add_right {n m : ℕ} (H : n ≤ m) (k : ℕ) : n + k ≤ m + k :=
|
|
|
|
|
!add.comm ▸ !add.comm ▸ nat.add_le_add_left H k
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem le_of_add_le_add_left {k n m : ℕ} (H : k + n ≤ k + m) : n ≤ m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
obtain l Hl, from le.elim H, le.intro (add.cancel_left (!add.assoc⁻¹ ⬝ Hl))
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem lt_of_add_lt_add_left {k n m : ℕ} (H : k + n < k + m) : n < m :=
|
2015-05-29 03:34:57 +00:00
|
|
|
|
let H' := le_of_lt H in
|
2015-10-22 19:31:58 +00:00
|
|
|
|
nat.lt_of_le_and_ne (nat.le_of_add_le_add_left H') (assume Heq, !lt.irrefl (Heq ▸ H))
|
2015-05-29 03:34:57 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem add_lt_add_left {n m : ℕ} (H : n < m) (k : ℕ) : k + n < k + m :=
|
|
|
|
|
lt_of_succ_le (!add_succ ▸ nat.add_le_add_left (succ_le_of_lt H) k)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem add_lt_add_right {n m : ℕ} (H : n < m) (k : ℕ) : n + k < m + k :=
|
|
|
|
|
!add.comm ▸ !add.comm ▸ nat.add_lt_add_left H k
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem lt_add_of_pos_right {n k : ℕ} (H : k > 0) : n < n + k :=
|
|
|
|
|
!add_zero ▸ nat.add_lt_add_left H n
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- multiplication -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-05-30 12:09:34 +00:00
|
|
|
|
theorem mul_le_mul_left {n m : ℕ} (k : ℕ) (H : n ≤ m) : k * n ≤ k * m :=
|
2015-01-07 01:44:04 +00:00
|
|
|
|
obtain (l : ℕ) (Hl : n + l = m), from le.elim H,
|
2015-10-14 19:27:09 +00:00
|
|
|
|
have k * n + k * l = k * m, by rewrite [-left_distrib, Hl],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
le.intro this
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-05-30 12:09:34 +00:00
|
|
|
|
theorem mul_le_mul_right {n m : ℕ} (k : ℕ) (H : n ≤ m) : n * k ≤ m * k :=
|
|
|
|
|
!mul.comm ▸ !mul.comm ▸ !mul_le_mul_left H
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem mul_le_mul {n m k l : ℕ} (H1 : n ≤ k) (H2 : m ≤ l) : n * m ≤ k * l :=
|
|
|
|
|
le.trans (!nat.mul_le_mul_right H1) (!nat.mul_le_mul_left H2)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem mul_lt_mul_of_pos_left {n m k : ℕ} (H : n < m) (Hk : k > 0) : k * n < k * m :=
|
|
|
|
|
lt_of_lt_of_le (nat.lt_add_of_pos_right Hk) (!mul_succ ▸ nat.mul_le_mul_left k (succ_le_of_lt H))
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem mul_lt_mul_of_pos_right {n m k : ℕ} (H : n < m) (Hk : k > 0) : n * k < m * k :=
|
|
|
|
|
!mul.comm ▸ !mul.comm ▸ nat.mul_lt_mul_of_pos_left H Hk
|
2015-07-20 02:47:14 +00:00
|
|
|
|
|
2015-08-04 02:41:37 +00:00
|
|
|
|
/- nat is an instance of a linearly ordered semiring and a lattice -/
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-12 03:35:45 +00:00
|
|
|
|
open algebra
|
2015-10-07 23:44:47 +00:00
|
|
|
|
|
2015-10-09 21:50:17 +00:00
|
|
|
|
protected definition decidable_linear_ordered_semiring [reducible] [trans_instance] :
|
2015-10-07 23:44:47 +00:00
|
|
|
|
algebra.decidable_linear_ordered_semiring nat :=
|
|
|
|
|
⦃ algebra.decidable_linear_ordered_semiring, nat.comm_semiring,
|
|
|
|
|
add_left_cancel := @add.cancel_left,
|
|
|
|
|
add_right_cancel := @add.cancel_right,
|
|
|
|
|
lt := nat.lt,
|
|
|
|
|
le := nat.le,
|
|
|
|
|
le_refl := le.refl,
|
|
|
|
|
le_trans := @le.trans,
|
|
|
|
|
le_antisymm := @le.antisymm,
|
|
|
|
|
le_total := @le.total,
|
2015-10-22 19:31:58 +00:00
|
|
|
|
le_iff_lt_or_eq := @nat.le_iff_lt_or_eq,
|
2015-10-07 23:44:47 +00:00
|
|
|
|
le_of_lt := @le_of_lt,
|
|
|
|
|
lt_irrefl := @lt.irrefl,
|
|
|
|
|
lt_of_lt_of_le := @lt_of_lt_of_le,
|
|
|
|
|
lt_of_le_of_lt := @lt_of_le_of_lt,
|
2015-10-22 19:31:58 +00:00
|
|
|
|
lt_of_add_lt_add_left := @nat.lt_of_add_lt_add_left,
|
|
|
|
|
add_lt_add_left := @nat.add_lt_add_left,
|
|
|
|
|
add_le_add_left := @nat.add_le_add_left,
|
|
|
|
|
le_of_add_le_add_left := @nat.le_of_add_le_add_left,
|
2015-10-07 23:44:47 +00:00
|
|
|
|
zero_lt_one := zero_lt_succ 0,
|
2015-10-22 19:31:58 +00:00
|
|
|
|
mul_le_mul_of_nonneg_left := (take a b c H1 H2, nat.mul_le_mul_left c H1),
|
|
|
|
|
mul_le_mul_of_nonneg_right := (take a b c H1 H2, nat.mul_le_mul_right c H1),
|
|
|
|
|
mul_lt_mul_of_pos_left := @nat.mul_lt_mul_of_pos_left,
|
|
|
|
|
mul_lt_mul_of_pos_right := @nat.mul_lt_mul_of_pos_right,
|
2015-10-07 23:44:47 +00:00
|
|
|
|
decidable_lt := nat.decidable_lt ⦄
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
definition nat_has_dvd [reducible] [instance] [priority nat.prio] : has_dvd nat :=
|
2015-10-09 19:47:55 +00:00
|
|
|
|
has_dvd.mk has_dvd.dvd
|
2015-10-07 23:44:47 +00:00
|
|
|
|
|
|
|
|
|
theorem add_pos_left {a : ℕ} (H : 0 < a) (b : ℕ) : 0 < a + b :=
|
|
|
|
|
@algebra.add_pos_of_pos_of_nonneg _ _ a b H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem add_pos_right {a : ℕ} (H : 0 < a) (b : ℕ) : 0 < b + a :=
|
|
|
|
|
by rewrite add.comm; apply add_pos_left H b
|
|
|
|
|
|
|
|
|
|
theorem add_eq_zero_iff_eq_zero_and_eq_zero {a b : ℕ} :
|
|
|
|
|
a + b = 0 ↔ a = 0 ∧ b = 0 :=
|
|
|
|
|
@algebra.add_eq_zero_iff_eq_zero_and_eq_zero_of_nonneg_of_nonneg _ _ a b !zero_le !zero_le
|
|
|
|
|
|
|
|
|
|
theorem le_add_of_le_left {a b c : ℕ} (H : b ≤ c) : b ≤ a + c :=
|
|
|
|
|
@algebra.le_add_of_nonneg_of_le _ _ a b c !zero_le H
|
|
|
|
|
|
|
|
|
|
theorem le_add_of_le_right {a b c : ℕ} (H : b ≤ c) : b ≤ c + a :=
|
|
|
|
|
@algebra.le_add_of_le_of_nonneg _ _ a b c H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem lt_add_of_lt_left {b c : ℕ} (H : b < c) (a : ℕ) : b < a + c :=
|
|
|
|
|
@algebra.lt_add_of_nonneg_of_lt _ _ a b c !zero_le H
|
|
|
|
|
|
|
|
|
|
theorem lt_add_of_lt_right {b c : ℕ} (H : b < c) (a : ℕ) : b < c + a :=
|
|
|
|
|
@algebra.lt_add_of_lt_of_nonneg _ _ a b c H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem lt_of_mul_lt_mul_left {a b c : ℕ} (H : c * a < c * b) : a < b :=
|
|
|
|
|
@algebra.lt_of_mul_lt_mul_left _ _ a b c H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem lt_of_mul_lt_mul_right {a b c : ℕ} (H : a * c < b * c) : a < b :=
|
|
|
|
|
@algebra.lt_of_mul_lt_mul_right _ _ a b c H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem pos_of_mul_pos_left {a b : ℕ} (H : 0 < a * b) : 0 < b :=
|
|
|
|
|
@algebra.pos_of_mul_pos_left _ _ a b H !zero_le
|
|
|
|
|
|
|
|
|
|
theorem pos_of_mul_pos_right {a b : ℕ} (H : 0 < a * b) : 0 < a :=
|
|
|
|
|
@algebra.pos_of_mul_pos_right _ _ a b H !zero_le
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-10-12 03:29:31 +00:00
|
|
|
|
theorem zero_le_one : (0:nat) ≤ 1 :=
|
|
|
|
|
dec_trivial
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
|
|
|
|
/- properties specific to nat -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
|
|
|
|
theorem lt_intro {n m k : ℕ} (H : succ n + k = m) : n < m :=
|
2015-01-07 01:44:04 +00:00
|
|
|
|
lt_of_succ_le (le.intro H)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem lt_elim {n m : ℕ} (H : n < m) : ∃k, succ n + k = m :=
|
|
|
|
|
le.elim (succ_le_of_lt H)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
|
|
|
|
theorem lt_add_succ (n m : ℕ) : n < n + succ m :=
|
2015-02-25 18:32:50 +00:00
|
|
|
|
lt_intro !succ_add_eq_succ_add
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem eq_zero_of_le_zero {n : ℕ} (H : n ≤ 0) : n = 0 :=
|
|
|
|
|
obtain (k : ℕ) (Hk : n + k = 0), from le.elim H,
|
|
|
|
|
eq_zero_of_add_eq_zero_right Hk
|
|
|
|
|
|
|
|
|
|
/- succ and pred -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem le_of_lt_succ {m n : nat} : m < succ n → m ≤ n :=
|
|
|
|
|
le_of_succ_le_succ
|
2015-06-04 23:57:56 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem lt_iff_succ_le (m n : nat) : m < n ↔ succ m ≤ n :=
|
2015-06-04 23:57:56 +00:00
|
|
|
|
iff.rfl
|
|
|
|
|
|
|
|
|
|
theorem lt_succ_iff_le (m n : nat) : m < succ n ↔ m ≤ n :=
|
|
|
|
|
iff.intro le_of_lt_succ lt_succ_of_le
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem self_le_succ (n : ℕ) : n ≤ succ n :=
|
|
|
|
|
le.intro !add_one
|
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem succ_le_or_eq_of_le {n m : ℕ} : n ≤ m → succ n ≤ m ∨ n = m :=
|
|
|
|
|
lt_or_eq_of_le
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
|
|
|
|
theorem pred_le_of_le_succ {n m : ℕ} : n ≤ succ m → pred n ≤ m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
pred_le_pred
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
|
|
|
|
theorem succ_le_of_le_pred {n m : ℕ} : succ n ≤ m → n ≤ pred m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
pred_le_pred
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
|
|
|
|
theorem pred_le_pred_of_le {n m : ℕ} : n ≤ m → pred n ≤ pred m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
pred_le_pred
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem pre_lt_of_lt {n m : ℕ} : n < m → pred n < m :=
|
|
|
|
|
lt_of_le_of_lt !pred_le
|
2015-06-03 05:08:25 +00:00
|
|
|
|
|
2015-01-12 21:28:42 +00:00
|
|
|
|
theorem lt_of_pred_lt_pred {n m : ℕ} (H : pred n < pred m) : n < m :=
|
2015-05-25 09:48:07 +00:00
|
|
|
|
lt_of_not_ge
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose m ≤ n,
|
|
|
|
|
not_lt_of_ge (pred_le_pred_of_le this) H)
|
2015-01-12 21:28:42 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem le_or_eq_succ_of_le_succ {n m : ℕ} (H : n ≤ succ m) : n ≤ m ∨ n = succ m :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
or.imp_left le_of_succ_le_succ (succ_le_or_eq_of_le H)
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
|
|
|
|
theorem le_pred_self (n : ℕ) : pred n ≤ n :=
|
2015-07-24 15:56:18 +00:00
|
|
|
|
!pred_le
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2014-11-22 08:15:51 +00:00
|
|
|
|
theorem succ_pos (n : ℕ) : 0 < succ n :=
|
|
|
|
|
!zero_lt_succ
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2014-12-17 18:32:38 +00:00
|
|
|
|
theorem succ_pred_of_pos {n : ℕ} (H : n > 0) : succ (pred n) = n :=
|
2015-01-07 01:44:04 +00:00
|
|
|
|
(or_resolve_right (eq_zero_or_eq_succ_pred n) (ne.symm (ne_of_lt H)))⁻¹
|
2014-12-17 18:32:38 +00:00
|
|
|
|
|
2015-07-24 15:56:18 +00:00
|
|
|
|
theorem exists_eq_succ_of_lt {n : ℕ} : Π {m : ℕ}, n < m → ∃k, m = succ k
|
|
|
|
|
| 0 H := absurd H !not_lt_zero
|
|
|
|
|
| (succ k) H := exists.intro k rfl
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-02-25 18:32:50 +00:00
|
|
|
|
theorem lt_succ_self (n : ℕ) : n < succ n :=
|
2014-11-22 08:15:51 +00:00
|
|
|
|
lt.base n
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-07-13 21:51:35 +00:00
|
|
|
|
lemma lt_succ_of_lt {i j : nat} : i < j → i < succ j :=
|
|
|
|
|
assume Plt, lt.trans Plt (self_lt_succ j)
|
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- other forms of induction -/
|
2015-07-31 13:09:18 +00:00
|
|
|
|
|
2015-07-15 19:49:47 +00:00
|
|
|
|
protected definition strong_rec_on {P : nat → Type} (n : ℕ) (H : ∀n, (∀m, m < n → P m) → P n) : P n :=
|
2015-08-10 03:47:41 +00:00
|
|
|
|
nat.rec (λm h, absurd h !not_lt_zero)
|
2015-07-24 15:56:18 +00:00
|
|
|
|
(λn' (IH : ∀ {m : ℕ}, m < n' → P m) m l,
|
|
|
|
|
or.by_cases (lt_or_eq_of_le (le_of_lt_succ l))
|
|
|
|
|
IH (λ e, eq.rec (H n' @IH) e⁻¹)) (succ n) n !lt_succ_self
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-07-15 19:49:47 +00:00
|
|
|
|
protected theorem strong_induction_on {P : nat → Prop} (n : ℕ) (H : ∀n, (∀m, m < n → P m) → P n) :
|
|
|
|
|
P n :=
|
|
|
|
|
nat.strong_rec_on n H
|
|
|
|
|
|
2014-09-19 22:04:52 +00:00
|
|
|
|
protected theorem case_strong_induction_on {P : nat → Prop} (a : nat) (H0 : P 0)
|
2014-08-02 02:29:50 +00:00
|
|
|
|
(Hind : ∀(n : nat), (∀m, m ≤ n → P m) → P (succ n)) : P a :=
|
2015-05-19 05:35:18 +00:00
|
|
|
|
nat.strong_induction_on a
|
|
|
|
|
(take n,
|
|
|
|
|
show (∀ m, m < n → P m) → P n, from
|
|
|
|
|
nat.cases_on n
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose (∀ m, m < 0 → P m), show P 0, from H0)
|
2014-08-02 02:29:50 +00:00
|
|
|
|
(take n,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
suppose (∀ m, m < succ n → P m),
|
2014-08-03 03:04:27 +00:00
|
|
|
|
show P (succ n), from
|
2015-07-22 20:41:50 +00:00
|
|
|
|
Hind n (take m, assume H1 : m ≤ n, this _ (lt_succ_of_le H1))))
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- pos -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-05-25 12:13:23 +00:00
|
|
|
|
theorem by_cases_zero_pos {P : ℕ → Prop} (y : ℕ) (H0 : P 0) (H1 : ∀ {y : nat}, y > 0 → P y) :
|
|
|
|
|
P y :=
|
2015-02-11 20:49:27 +00:00
|
|
|
|
nat.cases_on y H0 (take y, H1 !succ_pos)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-02-01 02:53:47 +00:00
|
|
|
|
theorem eq_zero_or_pos (n : ℕ) : n = 0 ∨ n > 0 :=
|
2014-12-15 20:05:44 +00:00
|
|
|
|
or_of_or_of_imp_left
|
2015-01-07 01:44:04 +00:00
|
|
|
|
(or.swap (lt_or_eq_of_le !zero_le))
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose 0 = n, by subst n)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem pos_of_ne_zero {n : ℕ} (H : n ≠ 0) : n > 0 :=
|
2015-05-26 01:14:52 +00:00
|
|
|
|
or.elim !eq_zero_or_pos (take H2 : n = 0, by contradiction) (take H2 : n > 0, H2)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem ne_zero_of_pos {n : ℕ} (H : n > 0) : n ≠ 0 :=
|
|
|
|
|
ne.symm (ne_of_lt H)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem exists_eq_succ_of_pos {n : ℕ} (H : n > 0) : exists l, n = succ l :=
|
|
|
|
|
exists_eq_succ_of_lt H
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem pos_of_dvd_of_pos {m n : ℕ} (H1 : m ∣ n) (H2 : n > 0) : m > 0 :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
pos_of_ne_zero
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose m = 0,
|
|
|
|
|
assert n = 0, from eq_zero_of_zero_dvd (this ▸ H1),
|
|
|
|
|
ne_of_lt H2 (by subst n))
|
2015-02-01 02:53:47 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
/- multiplication -/
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem mul_lt_mul_of_le_of_lt {n m k l : ℕ} (Hk : k > 0) (H1 : n ≤ k) (H2 : m < l) :
|
|
|
|
|
n * m < k * l :=
|
2015-05-30 12:09:34 +00:00
|
|
|
|
lt_of_le_of_lt (mul_le_mul_right m H1) (mul_lt_mul_of_pos_left H2 Hk)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem mul_lt_mul_of_lt_of_le {n m k l : ℕ} (Hl : l > 0) (H1 : n < k) (H2 : m ≤ l) :
|
|
|
|
|
n * m < k * l :=
|
2015-05-30 12:09:34 +00:00
|
|
|
|
lt_of_le_of_lt (mul_le_mul_left n H2) (mul_lt_mul_of_pos_right H1 Hl)
|
2014-08-02 02:29:50 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem mul_lt_mul_of_le_of_le {n m k l : ℕ} (H1 : n < k) (H2 : m < l) : n * m < k * l :=
|
2015-05-30 12:09:34 +00:00
|
|
|
|
have H3 : n * m ≤ k * m, from mul_le_mul_right m (le_of_lt H1),
|
2015-01-08 02:26:51 +00:00
|
|
|
|
have H4 : k * m < k * l, from mul_lt_mul_of_pos_left H2 (lt_of_le_of_lt !zero_le H1),
|
|
|
|
|
lt_of_le_of_lt H3 H4
|
2014-08-02 02:29:50 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem eq_of_mul_eq_mul_left {m k n : ℕ} (Hn : n > 0) (H : n * m = n * k) : m = k :=
|
2015-07-18 18:36:05 +00:00
|
|
|
|
have n * m ≤ n * k, by rewrite H,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
have m ≤ k, from le_of_mul_le_mul_left this Hn,
|
2015-07-18 18:36:05 +00:00
|
|
|
|
have n * k ≤ n * m, by rewrite H,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
have k ≤ m, from le_of_mul_le_mul_left this Hn,
|
|
|
|
|
le.antisymm `m ≤ k` this
|
2015-01-07 01:44:04 +00:00
|
|
|
|
|
2015-01-13 16:56:25 +00:00
|
|
|
|
theorem eq_of_mul_eq_mul_right {n m k : ℕ} (Hm : m > 0) (H : n * m = k * m) : n = k :=
|
|
|
|
|
eq_of_mul_eq_mul_left Hm (!mul.comm ▸ !mul.comm ▸ H)
|
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem eq_zero_or_eq_of_mul_eq_mul_left {n m k : ℕ} (H : n * m = n * k) : n = 0 ∨ m = k :=
|
2015-02-01 02:53:47 +00:00
|
|
|
|
or_of_or_of_imp_right !eq_zero_or_pos
|
2015-01-07 01:44:04 +00:00
|
|
|
|
(assume Hn : n > 0, eq_of_mul_eq_mul_left Hn H)
|
2014-08-02 02:29:50 +00:00
|
|
|
|
|
2015-01-07 01:44:04 +00:00
|
|
|
|
theorem eq_zero_or_eq_of_mul_eq_mul_right {n m k : ℕ} (H : n * m = k * m) : m = 0 ∨ n = k :=
|
|
|
|
|
eq_zero_or_eq_of_mul_eq_mul_left (!mul.comm ▸ !mul.comm ▸ H)
|
2014-08-02 02:29:50 +00:00
|
|
|
|
|
2015-01-13 16:56:25 +00:00
|
|
|
|
theorem eq_one_of_mul_eq_one_right {n m : ℕ} (H : n * m = 1) : n = 1 :=
|
2015-05-26 01:14:52 +00:00
|
|
|
|
have H2 : n * m > 0, by rewrite H; apply succ_pos,
|
2015-01-07 01:44:04 +00:00
|
|
|
|
or.elim (le_or_gt n 1)
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose n ≤ 1,
|
2015-07-18 18:36:05 +00:00
|
|
|
|
have n > 0, from pos_of_mul_pos_right H2,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
show n = 1, from le.antisymm `n ≤ 1` (succ_le_of_lt this))
|
|
|
|
|
(suppose n > 1,
|
2015-07-18 18:36:05 +00:00
|
|
|
|
have m > 0, from pos_of_mul_pos_left H2,
|
2015-10-22 19:31:58 +00:00
|
|
|
|
have n * m ≥ 2 * 1, from nat.mul_le_mul (succ_le_of_lt `n > 1`) (succ_le_of_lt this),
|
2015-07-18 18:36:05 +00:00
|
|
|
|
have 1 ≥ 2, from !mul_one ▸ H ▸ this,
|
|
|
|
|
absurd !lt_succ_self (not_lt_of_ge this))
|
2014-08-02 02:29:50 +00:00
|
|
|
|
|
2015-01-13 16:56:25 +00:00
|
|
|
|
theorem eq_one_of_mul_eq_one_left {n m : ℕ} (H : n * m = 1) : m = 1 :=
|
|
|
|
|
eq_one_of_mul_eq_one_right (!mul.comm ▸ H)
|
|
|
|
|
|
|
|
|
|
theorem eq_one_of_mul_eq_self_left {n m : ℕ} (Hpos : n > 0) (H : m * n = n) : m = 1 :=
|
|
|
|
|
eq_of_mul_eq_mul_right Hpos (H ⬝ !one_mul⁻¹)
|
|
|
|
|
|
|
|
|
|
theorem eq_one_of_mul_eq_self_right {n m : ℕ} (Hpos : m > 0) (H : m * n = m) : n = 1 :=
|
|
|
|
|
eq_one_of_mul_eq_self_left Hpos (!mul.comm ▸ H)
|
2014-08-02 01:40:24 +00:00
|
|
|
|
|
2015-04-06 16:24:09 +00:00
|
|
|
|
theorem eq_one_of_dvd_one {n : ℕ} (H : n ∣ 1) : n = 1 :=
|
2015-02-03 17:29:52 +00:00
|
|
|
|
dvd.elim H
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(take m, suppose 1 = n * m,
|
|
|
|
|
eq_one_of_mul_eq_one_right this⁻¹)
|
2015-02-03 17:29:52 +00:00
|
|
|
|
|
2015-06-29 02:09:31 +00:00
|
|
|
|
/- min and max -/
|
2015-07-20 02:47:14 +00:00
|
|
|
|
open decidable
|
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem min_zero [simp] (a : ℕ) : min a 0 = 0 :=
|
2015-07-20 02:47:14 +00:00
|
|
|
|
by rewrite [min_eq_right !zero_le]
|
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem zero_min [simp] (a : ℕ) : min 0 a = 0 :=
|
2015-07-20 02:47:14 +00:00
|
|
|
|
by rewrite [min_eq_left !zero_le]
|
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem max_zero [simp] (a : ℕ) : max a 0 = a :=
|
2015-07-20 02:47:14 +00:00
|
|
|
|
by rewrite [max_eq_left !zero_le]
|
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem zero_max [simp] (a : ℕ) : max 0 a = a :=
|
2015-07-20 02:47:14 +00:00
|
|
|
|
by rewrite [max_eq_right !zero_le]
|
2015-06-29 02:09:31 +00:00
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem min_succ_succ [simp] (a b : ℕ) : min (succ a) (succ b) = succ (min a b) :=
|
2015-08-04 02:41:37 +00:00
|
|
|
|
or.elim !lt_or_ge
|
|
|
|
|
(suppose a < b, by rewrite [min_eq_left_of_lt this, min_eq_left_of_lt (succ_lt_succ this)])
|
|
|
|
|
(suppose a ≥ b, by rewrite [min_eq_right this, min_eq_right (succ_le_succ this)])
|
2015-07-20 03:14:49 +00:00
|
|
|
|
|
2015-07-22 16:01:42 +00:00
|
|
|
|
theorem max_succ_succ [simp] (a b : ℕ) : max (succ a) (succ b) = succ (max a b) :=
|
2015-08-04 02:41:37 +00:00
|
|
|
|
or.elim !lt_or_ge
|
|
|
|
|
(suppose a < b, by rewrite [max_eq_right_of_lt this, max_eq_right_of_lt (succ_lt_succ this)])
|
|
|
|
|
(suppose a ≥ b, by rewrite [max_eq_left this, max_eq_left (succ_le_succ this)])
|
2015-06-29 02:09:31 +00:00
|
|
|
|
|
2015-08-04 02:41:37 +00:00
|
|
|
|
/- In algebra.ordered_group, these next four are only proved for additive groups, not additive
|
|
|
|
|
semigroups. -/
|
2015-06-29 02:09:31 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem min_add_add_left (a b c : ℕ) : min (a + b) (a + c) = a + min b c :=
|
2015-06-29 04:07:41 +00:00
|
|
|
|
decidable.by_cases
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose b ≤ c,
|
|
|
|
|
assert a + b ≤ a + c, from add_le_add_left this _,
|
|
|
|
|
by rewrite [min_eq_left `b ≤ c`, min_eq_left this])
|
|
|
|
|
(suppose ¬ b ≤ c,
|
|
|
|
|
assert c ≤ b, from le_of_lt (lt_of_not_ge this),
|
|
|
|
|
assert a + c ≤ a + b, from add_le_add_left this _,
|
|
|
|
|
by rewrite [min_eq_right `c ≤ b`, min_eq_right this])
|
2015-06-29 04:07:41 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem min_add_add_right (a b c : ℕ) : min (a + c) (b + c) = min a b + c :=
|
|
|
|
|
by rewrite [add.comm a c, add.comm b c, add.comm _ c]; apply nat.min_add_add_left
|
2015-06-29 04:07:41 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem max_add_add_left (a b c : ℕ) : max (a + b) (a + c) = a + max b c :=
|
2015-06-29 04:07:41 +00:00
|
|
|
|
decidable.by_cases
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose b ≤ c,
|
|
|
|
|
assert a + b ≤ a + c, from add_le_add_left this _,
|
|
|
|
|
by rewrite [max_eq_right `b ≤ c`, max_eq_right this])
|
|
|
|
|
(suppose ¬ b ≤ c,
|
|
|
|
|
assert c ≤ b, from le_of_lt (lt_of_not_ge this),
|
|
|
|
|
assert a + c ≤ a + b, from add_le_add_left this _,
|
|
|
|
|
by rewrite [max_eq_left `c ≤ b`, max_eq_left this])
|
2015-06-29 04:07:41 +00:00
|
|
|
|
|
2015-10-22 19:31:58 +00:00
|
|
|
|
protected theorem max_add_add_right (a b c : ℕ) : max (a + c) (b + c) = max a b + c :=
|
|
|
|
|
by rewrite [add.comm a c, add.comm b c, add.comm _ c]; apply nat.max_add_add_left
|
2015-06-29 04:07:41 +00:00
|
|
|
|
|
2015-08-04 00:49:20 +00:00
|
|
|
|
/- least and greatest -/
|
2015-07-07 23:54:21 +00:00
|
|
|
|
|
2015-08-04 00:49:20 +00:00
|
|
|
|
section least_and_greatest
|
2015-07-07 23:54:21 +00:00
|
|
|
|
variable (P : ℕ → Prop)
|
|
|
|
|
variable [decP : ∀ n, decidable (P n)]
|
|
|
|
|
include decP
|
|
|
|
|
|
2015-08-04 00:49:20 +00:00
|
|
|
|
-- returns the least i < n satisfying P, or n if there is none
|
|
|
|
|
definition least : ℕ → ℕ
|
|
|
|
|
| 0 := 0
|
|
|
|
|
| (succ n) := if P (least n) then least n else succ n
|
|
|
|
|
|
|
|
|
|
theorem least_of_bound {n : ℕ} (H : P n) : P (least P n) :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
rewrite ↑least,
|
|
|
|
|
apply H,
|
|
|
|
|
rewrite ↑least,
|
|
|
|
|
cases decidable.em (P (least P m)) with [Hlp, Hlp],
|
|
|
|
|
rewrite [if_pos Hlp],
|
|
|
|
|
apply Hlp,
|
|
|
|
|
rewrite [if_neg Hlp],
|
|
|
|
|
apply H
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem least_le (n : ℕ) : least P n ≤ n:=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
{rewrite ↑least},
|
|
|
|
|
rewrite ↑least,
|
|
|
|
|
cases decidable.em (P (least P m)) with [Psm, Pnsm],
|
|
|
|
|
rewrite [if_pos Psm],
|
|
|
|
|
apply le.trans ih !le_succ,
|
|
|
|
|
rewrite [if_neg Pnsm]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem least_of_lt {i n : ℕ} (ltin : i < n) (H : P i) : P (least P n) :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
exact absurd ltin !not_lt_zero,
|
|
|
|
|
rewrite ↑least,
|
|
|
|
|
cases decidable.em (P (least P m)) with [Psm, Pnsm],
|
|
|
|
|
rewrite [if_pos Psm],
|
|
|
|
|
apply Psm,
|
|
|
|
|
rewrite [if_neg Pnsm],
|
|
|
|
|
cases (lt_or_eq_of_le (le_of_lt_succ ltin)) with [Hlt, Heq],
|
|
|
|
|
exact absurd (ih Hlt) Pnsm,
|
|
|
|
|
rewrite Heq at H,
|
|
|
|
|
exact absurd (least_of_bound P H) Pnsm
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem ge_least_of_lt {i n : ℕ} (ltin : i < n) (Hi : P i) : i ≥ least P n :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
exact absurd ltin !not_lt_zero,
|
|
|
|
|
rewrite ↑least,
|
|
|
|
|
cases decidable.em (P (least P m)) with [Psm, Pnsm],
|
|
|
|
|
rewrite [if_pos Psm],
|
|
|
|
|
cases (lt_or_eq_of_le (le_of_lt_succ ltin)) with [Hlt, Heq],
|
|
|
|
|
apply ih Hlt,
|
|
|
|
|
rewrite Heq,
|
|
|
|
|
apply least_le,
|
|
|
|
|
rewrite [if_neg Pnsm],
|
|
|
|
|
cases (lt_or_eq_of_le (le_of_lt_succ ltin)) with [Hlt, Heq],
|
|
|
|
|
apply absurd (least_of_lt P Hlt Hi) Pnsm,
|
|
|
|
|
rewrite Heq at Hi,
|
|
|
|
|
apply absurd (least_of_bound P Hi) Pnsm
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem least_lt {n i : ℕ} (ltin : i < n) (Hi : P i) : least P n < n :=
|
|
|
|
|
lt_of_le_of_lt (ge_least_of_lt P ltin Hi) ltin
|
|
|
|
|
|
2015-07-07 23:54:21 +00:00
|
|
|
|
-- returns the largest i < n satisfying P, or n if there is none.
|
|
|
|
|
definition greatest : ℕ → ℕ
|
|
|
|
|
| 0 := 0
|
|
|
|
|
| (succ n) := if P n then n else greatest n
|
|
|
|
|
|
|
|
|
|
theorem greatest_of_lt {i n : ℕ} (ltin : i < n) (Hi : P i) : P (greatest P n) :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
{exact absurd ltin !not_lt_zero},
|
|
|
|
|
{cases (decidable.em (P m)) with [Psm, Pnsm],
|
|
|
|
|
{rewrite [↑greatest, if_pos Psm]; exact Psm},
|
|
|
|
|
{rewrite [↑greatest, if_neg Pnsm],
|
|
|
|
|
have neim : i ≠ m, from assume H : i = m, absurd (H ▸ Hi) Pnsm,
|
|
|
|
|
have ltim : i < m, from lt_of_le_of_ne (le_of_lt_succ ltin) neim,
|
|
|
|
|
apply ih ltim}}
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem le_greatest_of_lt {i n : ℕ} (ltin : i < n) (Hi : P i) : i ≤ greatest P n :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [m, ih],
|
|
|
|
|
{exact absurd ltin !not_lt_zero},
|
|
|
|
|
{cases (decidable.em (P m)) with [Psm, Pnsm],
|
|
|
|
|
{rewrite [↑greatest, if_pos Psm], apply le_of_lt_succ ltin},
|
|
|
|
|
{rewrite [↑greatest, if_neg Pnsm],
|
|
|
|
|
have neim : i ≠ m, from assume H : i = m, absurd (H ▸ Hi) Pnsm,
|
|
|
|
|
have ltim : i < m, from lt_of_le_of_ne (le_of_lt_succ ltin) neim,
|
|
|
|
|
apply ih ltim}}
|
2015-08-04 00:49:20 +00:00
|
|
|
|
end
|
2015-07-31 13:09:18 +00:00
|
|
|
|
|
2015-08-04 00:49:20 +00:00
|
|
|
|
end least_and_greatest
|
2015-07-07 23:54:21 +00:00
|
|
|
|
|
2014-08-20 02:32:44 +00:00
|
|
|
|
end nat
|