2015-04-14 15:59:01 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2015 Microsoft Corporation. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Authors: Leonardo de Moura
|
|
|
|
|
|
|
|
|
|
Elegant pairing function.
|
|
|
|
|
-/
|
|
|
|
|
import data.nat.sqrt data.nat.div
|
|
|
|
|
open prod decidable
|
|
|
|
|
|
|
|
|
|
namespace nat
|
|
|
|
|
definition mkpair (a b : nat) : nat :=
|
|
|
|
|
if a < b then b*b + a else a*a + a + b
|
|
|
|
|
|
|
|
|
|
definition unpair (n : nat) : nat × nat :=
|
|
|
|
|
let s := sqrt n in
|
|
|
|
|
if n - s*s < s then (n - s*s, s) else (s, n - s*s - s)
|
|
|
|
|
|
|
|
|
|
theorem mkpair_unpair (n : nat) : mkpair (pr1 (unpair n)) (pr2 (unpair n)) = n :=
|
|
|
|
|
let s := sqrt n in
|
|
|
|
|
by_cases
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose n - s*s < s,
|
2015-04-14 15:59:01 +00:00
|
|
|
|
begin
|
|
|
|
|
esimp [unpair],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
rewrite [if_pos this],
|
2015-04-14 15:59:01 +00:00
|
|
|
|
esimp [mkpair],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
rewrite [if_pos this, add_sub_of_le (sqrt_lower n)]
|
2015-04-14 15:59:01 +00:00
|
|
|
|
end)
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose h₁ : ¬ n - s*s < s,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have s ≤ n - s*s, from le_of_not_gt h₁,
|
|
|
|
|
have s + s*s ≤ n - s*s + s*s, from add_le_add_right this (s*s),
|
|
|
|
|
have s*s + s ≤ n, by rewrite [nat.sub_add_cancel (sqrt_lower n) at this,
|
2015-10-23 14:06:20 +00:00
|
|
|
|
add.comm at this]; assumption,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have n ≤ s*s + s + s, from sqrt_upper n,
|
|
|
|
|
have n - s*s ≤ s + s, from calc
|
2015-10-23 14:06:20 +00:00
|
|
|
|
n - s*s ≤ (s*s + s + s) - s*s : nat.sub_le_sub_right this (s*s)
|
2015-04-14 15:59:01 +00:00
|
|
|
|
... = (s*s + (s+s)) - s*s : by rewrite add.assoc
|
2015-10-23 14:06:20 +00:00
|
|
|
|
... = s + s : by rewrite nat.add_sub_cancel_left,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
have n - s*s - s ≤ s, from calc
|
2015-10-23 14:06:20 +00:00
|
|
|
|
n - s*s - s ≤ (s + s) - s : nat.sub_le_sub_right this s
|
|
|
|
|
... = s : by rewrite nat.add_sub_cancel_left,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have h₂ : ¬ s < n - s*s - s, from not_lt_of_ge this,
|
2015-04-14 15:59:01 +00:00
|
|
|
|
begin
|
|
|
|
|
esimp [unpair],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
rewrite [if_neg h₁], esimp,
|
2015-04-14 15:59:01 +00:00
|
|
|
|
esimp [mkpair],
|
2015-10-23 14:06:20 +00:00
|
|
|
|
rewrite [if_neg h₂, nat.sub_sub, add_sub_of_le `s*s + s ≤ n`],
|
2015-04-14 15:59:01 +00:00
|
|
|
|
end)
|
2015-04-15 03:28:20 +00:00
|
|
|
|
|
|
|
|
|
theorem unpair_mkpair (a b : nat) : unpair (mkpair a b) = (a, b) :=
|
|
|
|
|
by_cases
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose a < b,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have a ≤ b + b, from calc
|
2015-07-22 20:41:50 +00:00
|
|
|
|
a ≤ b : le_of_lt this
|
2015-04-15 03:28:20 +00:00
|
|
|
|
... ≤ b+b : !le_add_right,
|
|
|
|
|
begin
|
|
|
|
|
esimp [mkpair],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
rewrite [if_pos `a < b`],
|
2015-04-15 03:28:20 +00:00
|
|
|
|
esimp [unpair],
|
2015-10-23 14:06:20 +00:00
|
|
|
|
rewrite [sqrt_offset_eq `a ≤ b + b`, nat.add_sub_cancel_left, if_pos `a < b`]
|
2015-04-15 03:28:20 +00:00
|
|
|
|
end)
|
2015-07-22 20:41:50 +00:00
|
|
|
|
(suppose ¬ a < b,
|
|
|
|
|
have b ≤ a, from le_of_not_gt this,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have a + b ≤ a + a, from add_le_add_left this a,
|
2015-07-22 20:41:50 +00:00
|
|
|
|
have a + b ≥ a, from !le_add_right,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have ¬ a + b < a, from not_lt_of_ge this,
|
2015-04-15 03:28:20 +00:00
|
|
|
|
begin
|
|
|
|
|
esimp [mkpair],
|
2015-07-22 20:41:50 +00:00
|
|
|
|
rewrite [if_neg `¬ a < b`],
|
2015-04-15 03:28:20 +00:00
|
|
|
|
esimp [unpair],
|
2015-12-06 07:27:46 +00:00
|
|
|
|
rewrite [add.assoc (a * a) a b, sqrt_offset_eq `a + b ≤ a + a`, *nat.add_sub_cancel_left,
|
2015-10-23 14:06:20 +00:00
|
|
|
|
if_neg `¬ a + b < a`]
|
2015-04-15 03:28:20 +00:00
|
|
|
|
end)
|
2015-08-10 23:04:02 +00:00
|
|
|
|
|
|
|
|
|
open prod.ops
|
|
|
|
|
|
|
|
|
|
theorem unpair_lt_aux {n : nat} : n ≥ 1 → (unpair n).1 < n :=
|
|
|
|
|
suppose n ≥ 1,
|
|
|
|
|
or.elim (eq_or_lt_of_le this)
|
|
|
|
|
(suppose 1 = n, by subst n; exact dec_trivial)
|
|
|
|
|
(suppose n > 1,
|
|
|
|
|
let s := sqrt n in
|
|
|
|
|
by_cases
|
|
|
|
|
(suppose h : n - s*s < s,
|
2016-02-29 19:47:33 +00:00
|
|
|
|
have n > 0, from lt_of_succ_lt `n > 1`,
|
|
|
|
|
have sqrt n > 0, from sqrt_pos_of_pos this,
|
|
|
|
|
have sqrt n * sqrt n > 0, from mul_pos this this,
|
2015-08-10 23:04:02 +00:00
|
|
|
|
begin unfold unpair, rewrite [if_pos h], esimp, exact sub_lt `n > 0` `sqrt n * sqrt n > 0` end)
|
|
|
|
|
(suppose ¬ n - s*s < s, begin unfold unpair, rewrite [if_neg this], esimp, apply sqrt_lt `n > 1` end))
|
|
|
|
|
|
|
|
|
|
theorem unpair_lt : ∀ (n : nat), (unpair n).1 < succ n
|
|
|
|
|
| 0 := dec_trivial
|
|
|
|
|
| (succ n) :=
|
|
|
|
|
have (unpair (succ n)).1 < succ n, from unpair_lt_aux dec_trivial,
|
|
|
|
|
lt.step this
|
2015-04-14 15:59:01 +00:00
|
|
|
|
end nat
|