2015-07-11 01:01:06 +00:00
|
|
|
|
/-
|
|
|
|
|
Copyright (c) 2015 Jeremy Avigad. All rights reserved.
|
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
Author: Jeremy Avigad
|
|
|
|
|
|
|
|
|
|
Binomial coefficients, "n choose k".
|
|
|
|
|
-/
|
2015-07-27 03:24:35 +00:00
|
|
|
|
import data.nat.div data.nat.fact data.finset
|
2015-07-11 01:01:06 +00:00
|
|
|
|
open decidable
|
|
|
|
|
|
|
|
|
|
namespace nat
|
|
|
|
|
|
|
|
|
|
/- choose -/
|
|
|
|
|
|
|
|
|
|
definition choose : ℕ → ℕ → ℕ
|
|
|
|
|
| 0 0 := 1
|
|
|
|
|
| 0 (succ k) := 0
|
|
|
|
|
| (succ n) 0 := 1
|
|
|
|
|
| (succ n) (succ k) := choose n (succ k) + choose n k
|
|
|
|
|
|
|
|
|
|
theorem choose_zero_right (n : ℕ) : choose n 0 = 1 :=
|
|
|
|
|
nat.cases_on n rfl (take m, rfl)
|
|
|
|
|
|
|
|
|
|
theorem choose_zero_succ (k : ℕ) : choose 0 (succ k) = 0 := rfl
|
|
|
|
|
|
2015-07-27 03:24:35 +00:00
|
|
|
|
theorem choose_succ_succ (n k : ℕ) : choose (succ n) (succ k) = choose n (succ k) + choose n k :=
|
|
|
|
|
rfl
|
2015-07-11 01:01:06 +00:00
|
|
|
|
|
|
|
|
|
theorem choose_eq_zero_of_lt {n : ℕ} : ∀{k : ℕ}, n < k → choose n k = 0 :=
|
|
|
|
|
nat.induction_on n
|
|
|
|
|
(take k, assume H : 0 < k,
|
|
|
|
|
obtain k' (H : k = succ k'), from exists_eq_succ_of_pos H,
|
|
|
|
|
by rewrite H)
|
|
|
|
|
(take n',
|
|
|
|
|
assume IH: ∀ k, n' < k → choose n' k = 0,
|
|
|
|
|
take k,
|
2015-07-24 01:01:46 +00:00
|
|
|
|
suppose succ n' < k,
|
|
|
|
|
obtain k' (keq : k = succ k'), from exists_eq_succ_of_lt this,
|
|
|
|
|
assert n' < k', by rewrite keq at this; apply lt_of_succ_lt_succ this,
|
|
|
|
|
by rewrite [keq, choose_succ_succ, IH _ this, IH _ (lt.trans this !lt_succ_self)])
|
2015-07-11 01:01:06 +00:00
|
|
|
|
|
|
|
|
|
theorem choose_self (n : ℕ) : choose n n = 1 :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [n, ih],
|
|
|
|
|
{apply rfl},
|
|
|
|
|
rewrite [choose_succ_succ, ih, choose_eq_zero_of_lt !lt_succ_self]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem choose_succ_self (n : ℕ) : choose (succ n) n = succ n :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [n, ih],
|
|
|
|
|
{apply rfl},
|
|
|
|
|
rewrite [choose_succ_succ, ih, choose_self, add.comm]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem choose_one_right (n : ℕ) : choose n 1 = n :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [n, ih],
|
|
|
|
|
{apply rfl},
|
2015-10-12 03:29:31 +00:00
|
|
|
|
change choose (succ n) (succ 0) = succ n,
|
2016-02-04 21:15:42 +00:00
|
|
|
|
krewrite [choose_succ_succ, ih, choose_zero_right]
|
2015-07-11 01:01:06 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem choose_pos {n : ℕ} : ∀ {k : ℕ}, k ≤ n → choose n k > 0 :=
|
|
|
|
|
begin
|
|
|
|
|
induction n with [n, ih],
|
|
|
|
|
{intros [k, H],
|
2015-07-24 01:15:24 +00:00
|
|
|
|
have k = 0, from eq_of_le_of_ge H !zero_le,
|
|
|
|
|
subst k, rewrite choose_zero_right; apply zero_lt_one},
|
2015-07-11 01:01:06 +00:00
|
|
|
|
intro k,
|
|
|
|
|
cases k with k,
|
2015-07-24 01:15:24 +00:00
|
|
|
|
{intros, rewrite [choose_zero_right], apply zero_lt_one},
|
2015-07-24 01:01:46 +00:00
|
|
|
|
suppose succ k ≤ succ n,
|
|
|
|
|
assert k ≤ n, from le_of_succ_le_succ this,
|
|
|
|
|
by rewrite [choose_succ_succ]; apply add_pos_right (ih this)
|
2015-07-11 01:01:06 +00:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
-- A key identity. The proof is subtle.
|
2015-07-27 03:24:35 +00:00
|
|
|
|
theorem succ_mul_choose_eq (n : ℕ) :
|
|
|
|
|
∀ k, succ n * (choose n k) = choose (succ n) (succ k) * succ k :=
|
2015-07-11 01:01:06 +00:00
|
|
|
|
begin
|
|
|
|
|
induction n with [n, ih],
|
|
|
|
|
{intro k,
|
|
|
|
|
cases k with k',
|
|
|
|
|
{rewrite [*choose_self, one_mul, mul_one]},
|
|
|
|
|
{have H : 1 < succ (succ k'), from succ_lt_succ !zero_lt_succ,
|
2015-10-14 19:27:09 +00:00
|
|
|
|
krewrite [one_mul, choose_zero_succ, choose_eq_zero_of_lt H, zero_mul]}},
|
2015-07-11 01:01:06 +00:00
|
|
|
|
intro k,
|
|
|
|
|
cases k with k',
|
2016-02-04 21:15:42 +00:00
|
|
|
|
{krewrite [choose_zero_right, choose_one_right]},
|
2015-10-14 19:27:09 +00:00
|
|
|
|
rewrite [choose_succ_succ (succ n), right_distrib, -ih (succ k')],
|
|
|
|
|
rewrite [choose_succ_succ at {1}, left_distrib, *succ_mul (succ n), mul_succ, -ih k'],
|
2015-07-11 01:01:06 +00:00
|
|
|
|
rewrite [*add.assoc, add.left_comm (choose n _)]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
theorem choose_mul_fact_mul_fact {n : ℕ} :
|
|
|
|
|
∀ {k : ℕ}, k ≤ n → choose n k * fact k * fact (n - k) = fact n :=
|
|
|
|
|
begin
|
|
|
|
|
induction n using nat.strong_induction_on with [n, ih],
|
|
|
|
|
cases n with n,
|
2015-07-24 01:01:46 +00:00
|
|
|
|
{intro k H, have k = 0, from eq_zero_of_le_zero H, rewrite this},
|
2015-07-11 01:01:06 +00:00
|
|
|
|
intro k,
|
|
|
|
|
intro H, -- k ≤ n,
|
|
|
|
|
cases k with k,
|
|
|
|
|
{rewrite [choose_zero_right, fact_zero, *one_mul]},
|
2015-07-24 01:01:46 +00:00
|
|
|
|
have k ≤ n, from le_of_succ_le_succ H,
|
2015-07-11 01:01:06 +00:00
|
|
|
|
show choose (succ n) (succ k) * fact (succ k) * fact (succ n - succ k) = fact (succ n), from
|
|
|
|
|
begin
|
|
|
|
|
rewrite [succ_sub_succ, fact_succ, -mul.assoc, -succ_mul_choose_eq],
|
2015-07-24 01:01:46 +00:00
|
|
|
|
rewrite [fact_succ n, -ih n !lt_succ_self this, *mul.assoc]
|
2015-07-11 01:01:06 +00:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
2015-10-29 19:36:26 +00:00
|
|
|
|
theorem choose_def_alt {n k : ℕ} (H : k ≤ n) : choose n k = fact n / (fact k * fact (n -k)) :=
|
2015-10-23 14:06:20 +00:00
|
|
|
|
eq.symm (nat.div_eq_of_eq_mul_left (mul_pos !fact_pos !fact_pos)
|
2015-07-11 01:01:06 +00:00
|
|
|
|
(by rewrite [-mul.assoc, choose_mul_fact_mul_fact H]))
|
|
|
|
|
|
|
|
|
|
theorem fact_mul_fact_dvd_fact {n k : ℕ} (H : k ≤ n) : fact k * fact (n - k) ∣ fact n :=
|
|
|
|
|
by rewrite [-choose_mul_fact_mul_fact H, mul.assoc]; apply dvd_mul_left
|
|
|
|
|
|
2015-07-27 03:24:35 +00:00
|
|
|
|
open finset
|
|
|
|
|
|
|
|
|
|
/- the number of subsets of s of size k is n choose k -/
|
|
|
|
|
|
|
|
|
|
section card_subsets
|
|
|
|
|
variables {A : Type} [deceqA : decidable_eq A]
|
|
|
|
|
include deceqA
|
|
|
|
|
|
|
|
|
|
private theorem aux₀ (s : finset A) : {t ∈ powerset s | card t = 0} = '{∅} :=
|
|
|
|
|
ext (take t, iff.intro
|
|
|
|
|
(assume H,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
assert t = ∅, from eq_empty_of_card_eq_zero (of_mem_sep H),
|
2015-12-31 19:50:55 +00:00
|
|
|
|
show t ∈ '{ ∅ }, by rewrite [this, mem_singleton_iff])
|
2015-07-27 03:24:35 +00:00
|
|
|
|
(assume H,
|
2015-12-31 19:50:55 +00:00
|
|
|
|
assert t = ∅, by rewrite mem_singleton_iff at H; assumption,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
by substvars; exact mem_sep_of_mem !empty_mem_powerset rfl))
|
2015-07-27 03:24:35 +00:00
|
|
|
|
|
|
|
|
|
private theorem aux₁ (k : ℕ) : {t ∈ powerset (∅ : finset A) | card t = succ k} = ∅ :=
|
|
|
|
|
eq_empty_of_forall_not_mem (take t, assume H,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
assert t ∈ powerset ∅, from mem_of_mem_sep H,
|
2015-12-31 19:50:55 +00:00
|
|
|
|
assert t = ∅, by rewrite [powerset_empty at this, mem_singleton_iff at this]; assumption,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
have card (∅ : finset A) = succ k, by rewrite -this; apply of_mem_sep H,
|
2015-07-27 03:24:35 +00:00
|
|
|
|
nat.no_confusion this)
|
|
|
|
|
|
|
|
|
|
private theorem aux₂ {a : A} {s t : finset A} (anins : a ∉ s) (tpows : t ∈ powerset s) : a ∉ t :=
|
|
|
|
|
suppose a ∈ t,
|
|
|
|
|
have a ∈ s, from mem_of_subset_of_mem (subset_of_mem_powerset tpows) this,
|
|
|
|
|
anins this
|
|
|
|
|
|
|
|
|
|
private theorem aux₃ {a : A} {s t : finset A} (anins : a ∉ s) (k : ℕ) :
|
2016-01-04 00:41:01 +00:00
|
|
|
|
t ∈ (insert a) ' (powerset s) ∧ card t = succ k ↔
|
|
|
|
|
t ∈ (insert a) ' {t' ∈ powerset s | card t' = k} :=
|
2015-07-27 03:24:35 +00:00
|
|
|
|
iff.intro
|
|
|
|
|
(assume H,
|
|
|
|
|
obtain H' cardt, from H,
|
|
|
|
|
obtain t' [(t'pows : t' ∈ powerset s) (teq : insert a t' = t)], from exists_of_mem_image H',
|
|
|
|
|
assert aint : a ∈ t, by rewrite -teq; apply mem_insert,
|
|
|
|
|
assert anint' : a ∉ t', from
|
|
|
|
|
(assume aint',
|
|
|
|
|
have a ∈ s, from mem_of_subset_of_mem (subset_of_mem_powerset t'pows) aint',
|
|
|
|
|
anins this),
|
|
|
|
|
assert t' = erase a t, by rewrite [-teq, erase_insert (aux₂ anins t'pows)],
|
|
|
|
|
have card t' = k, by rewrite [this, card_erase_of_mem aint, cardt],
|
2015-08-08 22:10:44 +00:00
|
|
|
|
mem_image (mem_sep_of_mem t'pows this) teq)
|
2015-07-27 03:24:35 +00:00
|
|
|
|
(assume H,
|
|
|
|
|
obtain t' [Ht' (teq : insert a t' = t)], from exists_of_mem_image H,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
assert t'pows : t' ∈ powerset s, from mem_of_mem_sep Ht',
|
|
|
|
|
assert cardt' : card t' = k, from of_mem_sep Ht',
|
2015-07-27 03:24:35 +00:00
|
|
|
|
and.intro
|
2016-01-04 00:41:01 +00:00
|
|
|
|
(show t ∈ (insert a) ' (powerset s), from mem_image t'pows teq)
|
2015-07-27 03:24:35 +00:00
|
|
|
|
(show card t = succ k,
|
|
|
|
|
by rewrite [-teq, card_insert_of_not_mem (aux₂ anins t'pows), cardt']))
|
|
|
|
|
|
|
|
|
|
private theorem aux₄ {a : A} {s : finset A} (anins : a ∉ s) (k : ℕ) :
|
|
|
|
|
{t ∈ powerset (insert a s)| card t = succ k} =
|
2016-01-04 00:41:01 +00:00
|
|
|
|
{t ∈ powerset s | card t = succ k} ∪ (insert a) ' {t ∈ powerset s | card t = k} :=
|
2015-07-27 03:24:35 +00:00
|
|
|
|
begin
|
|
|
|
|
apply ext, intro t,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
rewrite [powerset_insert anins, mem_union_iff, *mem_sep_iff, mem_union_iff, and.right_distrib,
|
2015-07-27 03:24:35 +00:00
|
|
|
|
aux₃ anins]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
private theorem aux₅ {a : A} {s : finset A} (anins : a ∉ s) (k : ℕ) :
|
2016-01-04 00:41:01 +00:00
|
|
|
|
{t ∈ powerset s | card t = succ k} ∩ (insert a) ' {t ∈ powerset s | card t = k} = ∅ :=
|
2015-07-27 03:24:35 +00:00
|
|
|
|
inter_eq_empty
|
|
|
|
|
(take t, assume Ht₁ Ht₂,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
have tpows : t ∈ powerset s, from mem_of_mem_sep Ht₁,
|
2015-07-27 03:24:35 +00:00
|
|
|
|
have anint : a ∉ t, from aux₂ anins tpows,
|
|
|
|
|
obtain t' [Ht' (teq : insert a t' = t)], from exists_of_mem_image Ht₂,
|
|
|
|
|
have aint : a ∈ t, by rewrite -teq; apply mem_insert,
|
|
|
|
|
show false, from anint aint)
|
|
|
|
|
|
|
|
|
|
private theorem aux₆ {a : A} {s : finset A} (anins : a ∉ s) (k : ℕ) :
|
2016-01-04 00:41:01 +00:00
|
|
|
|
card ((insert a) ' {t ∈ powerset s | card t = k}) = card {t ∈ powerset s | card t = k} :=
|
2015-07-27 03:24:35 +00:00
|
|
|
|
have set.inj_on (insert a) (ts {t ∈ powerset s| card t = k}), from
|
|
|
|
|
take t₁ t₂, assume Ht₁ Ht₂,
|
|
|
|
|
assume Heq : insert a t₁ = insert a t₂,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
have t₁ ∈ powerset s, from mem_of_mem_sep Ht₁,
|
2015-07-27 03:24:35 +00:00
|
|
|
|
assert anint₁ : a ∉ t₁, from aux₂ anins this,
|
2015-08-08 22:10:44 +00:00
|
|
|
|
have t₂ ∈ powerset s, from mem_of_mem_sep Ht₂,
|
2015-07-27 03:24:35 +00:00
|
|
|
|
assert anint₂ : a ∉ t₂, from aux₂ anins this,
|
|
|
|
|
calc
|
|
|
|
|
t₁ = erase a (insert a t₁) : by rewrite (erase_insert anint₁)
|
|
|
|
|
... = erase a (insert a t₂) : Heq
|
|
|
|
|
... = t₂ : by rewrite (erase_insert anint₂),
|
|
|
|
|
card_image_eq_of_inj_on this
|
|
|
|
|
|
|
|
|
|
theorem card_subsets (s : finset A) : ∀k, card {t ∈ powerset s | card t = k} = choose (card s) k :=
|
|
|
|
|
begin
|
|
|
|
|
induction s with a s anins ih,
|
|
|
|
|
{intro k,
|
|
|
|
|
cases k with k,
|
|
|
|
|
{rewrite aux₀},
|
|
|
|
|
rewrite aux₁},
|
|
|
|
|
intro k,
|
|
|
|
|
cases k with k,
|
|
|
|
|
{rewrite [aux₀, choose_zero_right]},
|
|
|
|
|
rewrite [*(card_insert_of_not_mem anins), aux₄ anins, card_union_of_disjoint (aux₅ anins k),
|
|
|
|
|
aux₆ anins k, *ih]
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
end card_subsets
|
|
|
|
|
|
2015-07-11 01:01:06 +00:00
|
|
|
|
end nat
|