lean2/library/theories/measure_theory/extended_real.lean

413 lines
16 KiB
Text
Raw Normal View History

/-
Copyright (c) 2015 Jacob Gross. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Jacob Gross, Jeremy Avigad
Extended reals.
-/
import data.real
open real eq.ops classical
-- This is a hack, to get around the fact that the top level names are inaccessible when
-- defining these theorems in the ereal namespace. Is there a better way?
private definition zero_mul' := @zero_mul
private definition mul_zero' := @mul_zero
private definition neg_neg' := @neg_neg
noncomputable theory
inductive ereal : Type :=
| of_real : → ereal
| infty : ereal
| neginfty : ereal
attribute ereal.of_real [coercion]
notation `∞` := ereal.infty
notation `-∞` := ereal.neginfty
namespace ereal
protected definition prio := num.pred real.prio
/- arithmetic operations on the ereals -/
definition ereal_has_zero [instance] [priority ereal.prio] : has_zero ereal :=
has_zero.mk (of_real 0)
definition ereal_has_one [instance] [priority ereal.prio] : has_one ereal :=
has_one.mk (of_real 1)
protected definition add : ereal → ereal → ereal
| (of_real x) (of_real y) := of_real (x + y)
| ∞ _ := ∞
| _ ∞ := ∞
| -∞ _ := -∞
| _ -∞ := -∞
protected definition neg : ereal → ereal
| (of_real x) := of_real (-x)
| ∞ := -∞
| -∞ := ∞
private definition blow_up [reducible] : ereal → ereal
| (of_real x) := if x = 0 then of_real 0 else if x > 0 then ∞ else -∞
| ∞ := ∞
| -∞ := -∞
protected definition mul : ereal → ereal → ereal
| (of_real x) (of_real y) := of_real (x * y)
| ∞ a := blow_up a
| a ∞ := blow_up a
| -∞ a := ereal.neg (blow_up a)
| a -∞ := ereal.neg (blow_up a)
definition ereal_has_add [instance] [priority ereal.prio] : has_add ereal :=
has_add.mk ereal.add
definition ereal_has_neg [instance] [priority ereal.prio] : has_neg ereal :=
has_neg.mk ereal.neg
protected definition sub (u v : ereal) : ereal := u + -v
definition ereal_has_sub [instance] [priority ereal.prio] : has_sub ereal :=
has_sub.mk ereal.sub
definition ereal_has_mul [instance] [priority ereal.prio] : has_mul ereal :=
has_mul.mk ereal.mul
protected theorem zero_def : (0 : ereal) = of_real 0 := rfl
protected theorem one_def : (1 : ereal) = of_real 1 := rfl
protected theorem add_def (x y : ereal) : x + y = ereal.add x y := rfl
protected theorem neg_def (x : ereal) : -x = ereal.neg x := rfl
protected theorem sub_eq_add_neg (u v : ereal) : u - v = u + -v := rfl
protected theorem mul_def (x y : ereal) : x * y = ereal.mul x y := rfl
theorem of_real.inj {x y : real} (H : of_real x = of_real y) : x = y :=
ereal.no_confusion H (assume H1, H1)
abbreviation eq_of_of_real_eq_of_real := @of_real.inj
theorem of_real_add (x y : real) : of_real (x + y) = of_real x + of_real y := rfl
theorem of_real_mul (x y : real) : of_real (x * y) = of_real x * of_real y := rfl
theorem infty_ne_neg_infty : ∞ ≠ -∞ := ereal.no_confusion
theorem infty_ne_of_real (x : real) : ∞ ≠ of_real x := ereal.no_confusion
theorem neg_infty_ne_of_real (x : real) : -∞ ≠ of_real x := ereal.no_confusion
/- properties of the arithmetic operations -/
protected theorem add_comm : ∀ u v : ereal, u + v = v + u
| (of_real x) (of_real y) := congr_arg of_real !add.comm
| ∞ v := by rewrite[*ereal.add_def, ↑ereal.add]
| u ∞ := by rewrite[*ereal.add_def, ↑ereal.add]
| -∞ v := by rewrite[*ereal.add_def, ↑ereal.add]
| u -∞ := by rewrite[*ereal.add_def, ↑ereal.add]
theorem infty_add : ∀ u, ∞ + u = ∞
| (of_real x) := rfl
| ∞ := rfl
| -∞ := rfl
theorem add_infty : ∀ u, u + ∞ = ∞
| (of_real x) := rfl
| ∞ := rfl
| -∞ := rfl
protected theorem add_assoc : ∀ u v w : ereal, (u + v) + w = u + (v + w)
| (of_real x) (of_real y) (of_real z) := congr_arg of_real !add.assoc
| ∞ v w := by rewrite [*infty_add, *add_infty]
| u ∞ w := by rewrite [*infty_add, *add_infty, infty_add]
| u v ∞ := by rewrite [*infty_add, *add_infty]
| (of_real x) (of_real y) -∞ := by rewrite[*ereal.add_def, ↑ereal.add]
| (of_real x) -∞ (of_real z) := by rewrite[*ereal.add_def, ↑ereal.add]
| -∞ (of_real y) (of_real z) := by rewrite[*ereal.add_def, ↑ereal.add]
| (of_real x) -∞ -∞ := by rewrite[*ereal.add_def, ↑ereal.add]
| -∞ (of_real y) -∞ := rfl
| -∞ -∞ (of_real z) := by rewrite[*ereal.add_def, ↑ereal.add]
| -∞ -∞ -∞ := rfl
protected theorem zero_add : ∀ u : ereal, 0 + u = u
| (of_real x) := congr_arg of_real !real.zero_add
| ∞ := rfl
| -∞ := rfl
protected theorem add_zero : ∀ u : ereal, u + 0 = u :=
by intro u; rewrite [ereal.add_comm, ereal.zero_add]
protected theorem mul_comm : ∀ u v : ereal, u * v = v * u
| (of_real x) (of_real y) := congr_arg of_real !mul.comm
| ∞ a := by rewrite [*ereal.mul_def, ↑ereal.mul]
| a ∞ := by rewrite [*ereal.mul_def, ↑ereal.mul]
| -∞ a := by rewrite [*ereal.mul_def, ↑ereal.mul]
| a -∞ := by rewrite [*ereal.mul_def, ↑ereal.mul]
protected theorem neg_neg : ∀ u : ereal, -(-u) = u
| ∞ := rfl
| (of_real x) := by rewrite [*ereal.neg_def, ↑ereal.neg, ▸*,
(neg_neg' x)]
| -∞ := rfl
theorem neg_infty : -∞ = - ∞ := rfl
protected theorem neg_zero : -(0 : ereal) = 0 := rfl
theorem infty_mul_pos {x : real} (H : x > 0) : ∞ * x = ∞ :=
have H1 : x ≠ 0, from ne_of_gt H,
by+ rewrite [*ereal.mul_def, ↑ereal.mul, if_neg H1, if_pos H]
theorem pos_mul_infty {x : real} (H : x > 0) : x * ∞ = ∞ :=
by rewrite [ereal.mul_comm, infty_mul_pos H]
theorem infty_mul_neg {x : real} (H : x < 0) : ∞ * x = -∞ :=
have H1 : x ≠ 0, from ne_of_lt H,
have H2 : ¬ x > 0, from not_lt_of_gt H,
by+ rewrite [*ereal.mul_def, ↑ereal.mul, if_neg H1, if_neg H2]
theorem neg_mul_infty {x : real} (H : x < 0) : x * ∞ = -∞ :=
by rewrite [ereal.mul_comm, infty_mul_neg H]
private theorem infty_mul_zero : ∞ * 0 = 0 :=
by rewrite [*ereal.mul_def, ↑ereal.mul, ereal.zero_def, ↑blow_up, if_pos rfl]
private theorem zero_mul_infty : 0 * ∞ = 0 :=
by rewrite [ereal.mul_comm, infty_mul_zero]
theorem infty_mul_infty : ∞ * ∞ = ∞ := rfl
protected theorem neg_of_real (x : real) : -(of_real x) = of_real (-x) :=
rfl
private theorem aux1 : ∀ v : ereal, -∞ * v = -(∞ * v)
| ∞ := rfl
| (of_real x) := rfl
| -∞ := rfl
private theorem aux2 : ∀ u : ereal, -u * ∞ = -(u * ∞)
| ∞ := rfl
| (of_real x) := lt.by_cases
(assume H : x < 0,
by rewrite [ereal.neg_of_real, pos_mul_infty (neg_pos_of_neg H),
neg_mul_infty H])
(assume H : x = 0,
by krewrite [H, ereal.neg_zero, *zero_mul_infty, ereal.neg_zero])
(assume H : x > 0,
by rewrite [ereal.neg_of_real, neg_mul_infty (neg_neg_of_pos H),
pos_mul_infty H])
| -∞ := rfl
theorem ereal_neg_mul : ∀ u v : ereal, -u * v = -(u * v)
| ∞ v := aux1 v
| -∞ v := by rewrite [aux1, *ereal.neg_neg]
| u ∞ := by rewrite [-aux2]
| u -∞ := by rewrite [ereal.mul_comm, ereal.mul_comm u,
*aux1, ereal.mul_comm, aux2, *ereal.neg_neg]
| (of_real x) (of_real y) := congr_arg of_real (eq.symm (neg_mul_eq_neg_mul x y))
theorem ereal_mul_neg (u v : ereal) : u * -v = -(u * v) :=
by rewrite [*ereal.mul_comm u, ereal_neg_mul]
protected theorem mul_zero : ∀ u : ereal, u * 0 = 0
| ∞ := infty_mul_zero
| -∞ := by rewrite [neg_infty, ereal_neg_mul, infty_mul_zero]
| (of_real x) := congr_arg of_real (mul_zero' x)
protected theorem zero_mul (u : ereal) : 0 * u = 0 :=
by rewrite [ereal.mul_comm, ereal.mul_zero]
private theorem aux3 : ∀ u, ∞ * (∞ * u) = ∞ * u
| ∞ := rfl
| (of_real x) := if H : x = 0 then
by rewrite [*ereal.mul_def, ↑ereal.mul, ↑blow_up, *H, *if_pos rfl]
else if H1 : x > 0 then
by rewrite [*ereal.mul_def, ↑ereal.mul, ↑blow_up, if_neg H, if_pos H1]
else
by rewrite [*ereal.mul_def, ↑ereal.mul, ↑blow_up, if_neg H, if_neg H1]
| -∞ := rfl
private theorem aux4 (x y : real) : ∞ * x * y = ∞ * (x * y) :=
lt.by_cases
(assume H : x < 0,
lt.by_cases
(assume H1 : y < 0, by rewrite [infty_mul_neg H, neg_infty, ereal_neg_mul, -of_real_mul,
infty_mul_neg H1, infty_mul_pos (mul_pos_of_neg_of_neg H H1)])
(assume H1 : y = 0, by krewrite [H1, *ereal.mul_zero])
(assume H1 : y > 0, by rewrite [infty_mul_neg H, neg_infty, *ereal_neg_mul, -of_real_mul,
infty_mul_pos H1, infty_mul_neg (mul_neg_of_neg_of_pos H H1)]))
(assume H : x = 0,
by krewrite [H, ereal.mul_zero, *ereal.zero_mul, ereal.mul_zero])
(assume H : x > 0,
lt.by_cases
(assume H1 : y < 0, by rewrite [infty_mul_pos H, infty_mul_neg H1, -of_real_mul,
infty_mul_neg (mul_neg_of_pos_of_neg H H1)])
(assume H1 : y = 0, by krewrite [H1, *ereal.mul_zero])
(assume H1 : y > 0, by rewrite [infty_mul_pos H, infty_mul_pos H1, -of_real_mul,
infty_mul_pos (mul_pos H H1)]))
private theorem aux5 : ∀ u v, ∞ * u * v = ∞ * (u * v)
| ∞ v := by rewrite [infty_mul_infty, aux3]
| u ∞ := by rewrite [-*ereal.mul_comm ∞]
| -∞ v := by rewrite [neg_infty, *ereal_neg_mul, *ereal_mul_neg, ereal_neg_mul, infty_mul_infty,
aux3]
| u -∞ := by rewrite [neg_infty, *ereal_mul_neg]
| (of_real x) (of_real y) := aux4 x y
protected theorem mul_assoc : ∀ u v w : ereal, u * v * w = u * (v * w)
| ∞ v w := !aux5
| u ∞ w := by rewrite [-*ereal.mul_comm ∞, *ereal.mul_comm u, *aux5, *ereal.mul_comm u]
| u v ∞ := by rewrite [-*ereal.mul_comm ∞, *ereal.mul_comm u, aux5]
| -∞ v w := by rewrite [neg_infty, *ereal_neg_mul, aux5]
| u -∞ w := by rewrite [neg_infty, *ereal_mul_neg, *ereal_neg_mul, ereal_mul_neg, *ereal.mul_comm u,
*aux5, ereal.mul_comm u]
| u v -∞ := by rewrite [neg_infty, *ereal_mul_neg, *ereal.mul_comm u, -*ereal.mul_comm ∞, aux5]
| (of_real x) (of_real y) (of_real z) := congr_arg of_real (mul.assoc x y z)
protected theorem one_mul : ∀ u : ereal, of_real 1 * u = u
| (of_real x) := !real.one_mul ▸ rfl
| ∞ := pos_mul_infty zero_lt_one
| -∞ := by rewrite [neg_infty, ereal_mul_neg, pos_mul_infty zero_lt_one]
protected theorem mul_one (u : ereal) : u * 1 = u :=
by krewrite [ereal.mul_comm, ereal.one_mul]
/- instantiating arithmetic structures -/
-- Note that distributivity fails, e.g. ∞ ⬝ (-1 + 1) ≠ ∞ * -1 + ∞ * 1
protected definition comm_monoid [reducible] [trans_instance] :
comm_monoid ereal :=
⦃comm_monoid,
mul := ereal.mul,
mul_assoc := ereal.mul_assoc,
one := 1,
one_mul := ereal.one_mul,
mul_one := ereal.mul_one,
mul_comm := ereal.mul_comm
protected definition add_comm_monoid [reducible] [trans_instance] :
add_comm_monoid ereal :=
⦃add_comm_monoid,
add := ereal.add,
add_assoc := ereal.add_assoc,
zero := 0,
zero_add := ereal.zero_add,
add_zero := ereal.add_zero,
add_comm := ereal.add_comm
/- ordering on the ereals -/
protected definition le : ereal → ereal → Prop
| u ∞ := true
| -∞ v := true
| (of_real x) (of_real y) := x ≤ y
| (of_real x) -∞ := false
| ∞ (of_real y) := false
| ∞ -∞ := false
definition ereal_has_le [instance] [priority ereal.prio] :
has_le ereal :=
has_le.mk ereal.le
theorem of_real_le_of_real (x y : real) : of_real x ≤ of_real y ↔ x ≤ y :=
!iff.refl
theorem le_infty : ∀ u, u ≤ ∞
| ∞ := trivial
| (of_real x) := trivial
| -∞ := trivial
theorem neg_infty_le : ∀ v, -∞ ≤ v
| ∞ := trivial
| (of_real x) := trivial
| -∞ := trivial
protected theorem le_refl : ∀ u : ereal, u ≤ u
| ∞ := trivial
| -∞ := trivial
| (of_real x) := by rewrite [of_real_le_of_real]; apply le.refl
protected theorem le_trans : ∀ u v w : ereal, u ≤ v → v ≤ w → u ≤ w
| u v ∞ H1 H2 := !le_infty
| -∞ v w H1 H2 := !neg_infty_le
| u ∞ (of_real x) H1 H2 := false.elim H2
| ∞ (of_real x) v H1 H2 := false.elim H1
| ∞ -∞ v H1 H2 := false.elim H1
| u (of_real x) -∞ H1 H2 := false.elim H2
| u ∞ -∞ H1 H2 := false.elim H2
| (of_real x) -∞ v H1 H2 := false.elim H1
| (of_real x) (of_real y) (of_real z) H1 H2 :=
iff.mpr !of_real_le_of_real
(le.trans (iff.mp !of_real_le_of_real H1) (iff.mp !of_real_le_of_real H2))
protected theorem le_antisymm : ∀ u v : ereal, u ≤ v → v ≤ u → u = v
| ∞ ∞ H1 H2 := rfl
| ∞ (of_real x) H1 H2 := false.elim H1
| ∞ -∞ H1 H2 := false.elim H1
| -∞ -∞ H1 H2 := rfl
| -∞ (of_real x) H1 H2 := false.elim H2
| -∞ ∞ H1 H2 := false.elim H2
| (of_real x) ∞ H1 H2 := false.elim H2
| (of_real x) -∞ H1 H2 := false.elim H1
| (of_real x) (of_real y) H1 H2 :=
congr_arg of_real (le.antisymm (iff.mp !of_real_le_of_real H1) (iff.mp !of_real_le_of_real H2))
protected definition lt (x y : ereal) : Prop := x ≤ y ∧ x ≠ y
definition ereal_has_lt [instance] [priority ereal.prio] :
has_lt ereal :=
has_lt.mk ereal.lt
protected theorem le_iff_lt_or_eq (u v : ereal) : u ≤ v ↔ u < v u = v :=
iff.intro
(assume H : u ≤ v,
by_cases
(assume H1 : u = v, or.inr H1)
(assume H1 : u ≠ v, or.inl (and.intro H H1)))
(assume H : u < v u = v,
or.elim H
(assume H1 : u < v, and.left H1)
(assume H1 : u = v, by rewrite H1; apply ereal.le_refl))
protected theorem le_total : ∀ u v : ereal, u ≤ v v ≤ u
| u ∞ := or.inl (le_infty u)
| u -∞ := or.inr (neg_infty_le u)
| ∞ v := or.inr (le_infty v)
| -∞ v := or.inl (neg_infty_le v)
| (of_real x) (of_real y) :=
or.elim (le.total x y)
(assume H : x ≤[real] y, or.inl (iff.mpr !of_real_le_of_real H))
(assume H : x ≥[real] y, or.inr (iff.mpr !of_real_le_of_real H))
theorem neg_infty_lt_infty : -∞ < ∞ := and.intro trivial (ne.symm infty_ne_neg_infty)
theorem neg_infty_lt_of_real (x : real) : -∞ < of_real x := and.intro trivial !neg_infty_ne_of_real
theorem of_real_lt_infty (x : real) : of_real x < ∞ := and.intro trivial (ne.symm !infty_ne_of_real)
protected definition decidable_linear_order [reducible] [trans_instance] :
decidable_linear_order ereal :=
⦃decidable_linear_order,
le := ereal.le,
le_refl := ereal.le_refl,
le_trans := ereal.le_trans,
le_antisymm := ereal.le_antisymm,
lt := ereal.lt,
le_iff_lt_or_eq := ereal.le_iff_lt_or_eq,
lt_irrefl := abstract λ u H, and.right H rfl end,
decidable_lt := abstract λ u v : ereal, prop_decidable (u < v) end,
le_total := ereal.le_total
-- TODO : we still need some properties relating the arithmetic operations and the order.
end ereal