FinMatrix.CoqExt.Hierarchy
Require Import Coq.Logic.Description. Require Export Basic.
Require Export Reals Ring Field.
Import ListNotations.
Open Scope nat_scope.
Open Scope A_scope.
Set Implicit Arguments.
Unset Strict Implicit.
Generalizable Variables tA Aadd Azero Aopp Amul Aone Ainv Adiv Alt.
Ltac split_intro :=
repeat (try split; try intro).
repeat (try split; try intro).
Applicate a unary function for n-times, i.e. f ( .. (f a0) ...)
Fixpoint iterate {tA} (f : tA -> tA) (n : nat) (a0 : tA) : tA :=
match n with
| O => a0
| S n' => f (iterate f n' a0)
end.
Section test.
Context {tA} {f : tA -> tA} (Azero : tA).
End test.
match n with
| O => a0
| S n' => f (iterate f n' a0)
end.
Section test.
Context {tA} {f : tA -> tA} (Azero : tA).
End test.
Class Dec {tA} (Acmp : tA -> tA -> Prop) := {
dec : forall (a b : tA), {Acmp a b} + {~(Acmp a b)};
}.
Arguments dec {tA} _ {_} : simpl never.
Definition Aeqdec {tA} {AeqDec:Dec (@eq tA)} := @dec _ (@eq tA) AeqDec.
Section Instances.
Import Arith ZArith Reals.
#[export] Instance Dec_list `{Dec _ (@eq tA)} : Dec (@eq (list tA)).
Proof. constructor. intros. apply list_eq_dec. apply Aeqdec. Defined.
Import Arith ZArith Reals.
#[export] Instance Dec_list `{Dec _ (@eq tA)} : Dec (@eq (list tA)).
Proof. constructor. intros. apply list_eq_dec. apply Aeqdec. Defined.
Equality of two pairs, iff their corresponding components are all equal.
Lemma prod_eq_iff : forall {tA tB} (z1 z2 : tA * tB),
z1 = z2 <-> fst z1 = fst z2 /\ snd z1 = snd z2.
Proof.
intros tA tB (a1,b1) (a2,b2). split; intros H; inv H; auto.
simpl in *; subst. auto.
Qed.
z1 = z2 <-> fst z1 = fst z2 /\ snd z1 = snd z2.
Proof.
intros tA tB (a1,b1) (a2,b2). split; intros H; inv H; auto.
simpl in *; subst. auto.
Qed.
Inequality of two pairs, iff at least one of components are not equal.
Lemma prod_neq_iff : forall {tA tB} {AeqDec:Dec (@eq tA)} {BeqDec:Dec (@eq tB)}
(z1 z2 : tA * tB),
z1 <> z2 <-> fst z1 <> fst z2 \/ snd z1 <> snd z2.
Proof.
intros. rewrite prod_eq_iff. split; intros.
apply not_and_or in H; auto.
apply or_not_and in H; auto.
Qed.
End Instances.
(z1 z2 : tA * tB),
z1 <> z2 <-> fst z1 <> fst z2 \/ snd z1 <> snd z2.
Proof.
intros. rewrite prod_eq_iff. split; intros.
apply not_and_or in H; auto.
apply or_not_and in H; auto.
Qed.
End Instances.
Respect: an operation respect a relation (also known as "well-defined")
Class Injective {tA tB} (phi: tA -> tB) := {
injective : forall a1 a2 : tA, a1 <> a2 -> phi a1 <> phi a2
}.
Second form of injective
These two forms are equal
Lemma injective_eq_injective_form2 (phi: tA -> tB) :
Injective phi <-> injective_form2 phi.
Proof.
split; intros.
- hnf. destruct H as [H]. intros.
specialize (H a1 a2). apply imply_to_or in H. destruct H.
+ apply NNPP in H. auto.
+ easy.
- hnf in H. constructor. intros. intro. apply H in H1. easy.
Qed.
Injective phi <-> injective_form2 phi.
Proof.
split; intros.
- hnf. destruct H as [H]. intros.
specialize (H a1 a2). apply imply_to_or in H. destruct H.
+ apply NNPP in H. auto.
+ easy.
- hnf in H. constructor. intros. intro. apply H in H1. easy.
Qed.
Injective function preserve equal relation
Lemma injective_preserve_eq : forall (f : tA -> tB),
Injective f -> (forall a1 a2, f a1 = f a2 -> a1 = a2).
Proof.
intros. apply injective_eq_injective_form2 in H. apply H. auto.
Qed.
End theory.
Injective f -> (forall a1 a2, f a1 = f a2 -> a1 = a2).
Proof.
intros. apply injective_eq_injective_form2 in H. apply H. auto.
Qed.
End theory.
Class Bijective {tA tB} (phi: tA -> tB) := {
bijInjective :: Injective phi;
bijSurjective :: Surjective phi
}.
Coercion bijInjective : Bijective >-> Injective.
Coercion bijSurjective : Bijective >-> Surjective.
There exist inverse function from a bijective function.
ref: https://stackoverflow.com/questions/62464821/
how-to-make-an-inverse-function-in-coq
Tips: there are two methods to formalize "existential", sig and ex.
ex makes a Prop, sig makes a Type.
Here, we proof the ex version. the sig version could be derived by an axiom:
constructive_definite_description :
forall (A : Type) (P : tA -> Prop), (exists ! x : tA, P x) -> {x : tA | P x}
Lemma bij_inverse_exist : forall (phi : tA -> tB) (Hbij: Bijective phi),
{psi : tB -> tA | (forall a : tA, psi (phi a) = a) /\ (forall b : tB, phi (psi b) = b)}.
Proof.
intros. destruct Hbij as [Hinj [Hsurj]].
apply injective_eq_injective_form2 in Hinj. hnf in *.
assert (H : forall b, exists! a, phi a = b).
{ intros b.
destruct (Hsurj b) as [a Ha]. exists a. repeat split; auto.
intros a' Ha'. apply Hinj. rewrite Ha. auto. }
apply constructive_definite_description.
exists (fun b => proj1_sig (constructive_definite_description _ (H b))). split.
- split.
+ intros a. destruct (constructive_definite_description). simpl. auto.
+ intros b. destruct (constructive_definite_description). simpl. auto.
- intro psi; intros. apply functional_extensionality.
intros b. destruct (constructive_definite_description). simpl.
destruct H0. rewrite <- e. auto.
Defined.
bijective function preserve equal relation
Lemma bijective_preserve_eq : forall (f : tA -> tB),
Bijective f -> (forall (a1 a2 : tA), f a1 = f a2 -> a1 = a2).
Proof.
intros. destruct H as [Hinj Hsurj].
apply injective_preserve_eq in H0; auto.
Qed.
End theory.
Bijective f -> (forall (a1 a2 : tA), f a1 = f a2 -> a1 = a2).
Proof.
intros. destruct H as [Hinj Hsurj].
apply injective_preserve_eq in H0; auto.
Qed.
End theory.
Class Homomorphic {tA tB}
(fa : tA -> tA -> tA) (fb : tB -> tB -> tB) (phi: tA -> tB) := {
homomorphic : forall (a1 a2 : tA), phi (fa a1 a2) = fb (phi a1) (phi a2)
}.
Homomorphism
Class
Class Homomorphism {tA tB} (fa : tA -> tA -> tA) (fb : tB -> tB -> tB) := {
homomorphism : exists (phi: tA -> tB), Homomorphic fa fb phi /\ Surjective phi
}.
homomorphism : exists (phi: tA -> tB), Homomorphic fa fb phi /\ Surjective phi
}.
If there exist two homomorphic and surjective mapping from <A,+> to <B,⊕>
and from <A,*> to <B,⊗>, then we said <A,+,*> and <B,⊕,⊗> is homomorphism
Class Homomorphism2 {tA tB} (fa ga : tA -> tA -> tA) (fb gb : tB -> tB -> tB) := {
homomorphism2 : exists (phi: tA -> tB),
Homomorphic fa fb phi /\ Homomorphic ga gb phi /\ Surjective phi
}.
homomorphism2 : exists (phi: tA -> tB),
Homomorphic fa fb phi /\ Homomorphic ga gb phi /\ Surjective phi
}.
Isomorphism
Class
Class Isomorphism {tA tB} (fa : tA -> tA -> tA) (fb : tB -> tB -> tB) := {
isomorphism : exists (phi: tA -> tB), Homomorphic fa fb phi /\ Bijective phi
}.
isomorphism : exists (phi: tA -> tB), Homomorphic fa fb phi /\ Bijective phi
}.
If there exist two homomorphic and bijective mapping from <A,+> to <B,⊕>
and from <A,*> to <B,⊗>, then we said <A,+,*> and <B,⊕,⊗> is isomorphism
Class Isomorphism2 {tA tB} (fa ga : tA -> tA -> tA) (fb gb : tB -> tB -> tB) := {
isomorphism2 : exists (phi: tA -> tB),
Homomorphic fa fb phi /\ Homomorphic ga gb phi /\ Bijective phi
}.
isomorphism2 : exists (phi: tA -> tB),
Homomorphic fa fb phi /\ Homomorphic ga gb phi /\ Bijective phi
}.
Class Associative {tA} (Aop : tA -> tA -> tA) := {
associative : forall a b c, Aop (Aop a b) c = Aop a (Aop b c);
}.
associative : forall a b c, Aop (Aop a b) c = Aop a (Aop b c);
}.
Goal forall a b c : nat, (a + (b + c) = (a + b) + c).
intros. rewrite associative; auto. Qed.
Goal forall a b c : nat, ((a + b) + c = a + (b + c)).
apply associative. Qed.
intros. rewrite associative; auto. Qed.
Goal forall a b c : nat, ((a + b) + c = a + (b + c)).
apply associative. Qed.
#[export] Instance Comm_NatAdd : Commutative Nat.add.
constructor. auto with arith. Defined.
#[export] Instance Comm_NatMul : Commutative Nat.mul.
constructor. auto with arith. Defined.
constructor. auto with arith. Defined.
#[export] Instance Comm_NatMul : Commutative Nat.mul.
constructor. auto with arith. Defined.
Goal forall a b : nat, (a + b = b + a)%nat.
apply commutative. Qed.
Goal forall a b : nat, (a * b = b * a)%nat.
apply commutative. Qed.
apply commutative. Qed.
Goal forall a b : nat, (a * b = b * a)%nat.
apply commutative. Qed.
Class IdentityLeft {tA} (Aop : tA -> tA -> tA) (Ae : tA) := {
identityLeft : forall a, Aop Ae a = a
}.
Class IdentityRight {tA} (Aop : tA -> tA -> tA) (Ae : tA) := {
identityRight : forall a, Aop a Ae = a
}.
identityLeft : forall a, Aop Ae a = a
}.
Class IdentityRight {tA} (Aop : tA -> tA -> tA) (Ae : tA) := {
identityRight : forall a, Aop a Ae = a
}.
Class InverseLeft {tA} (Aop : tA -> tA -> tA) (Ae : tA) (Aopinv : tA -> tA)
:= {
inverseLeft : forall a, Aop (Aopinv a) a = Ae
}.
Class InverseRight {tA} (Aop : tA -> tA -> tA) (Ae : tA) (Aopinv : tA -> tA)
:= {
inverseRight : forall a, Aop a (Aopinv a) = Ae
}.
:= {
inverseLeft : forall a, Aop (Aopinv a) a = Ae
}.
Class InverseRight {tA} (Aop : tA -> tA -> tA) (Ae : tA) (Aopinv : tA -> tA)
:= {
inverseRight : forall a, Aop a (Aopinv a) = Ae
}.
Class DistrLeft {tA} (Aadd Amul : tA -> tA -> tA) := {
distrLeft : forall a b c,
Amul a (Aadd b c) = Aadd (Amul a b) (Amul a c)
}.
Class DistrRight {tA} (Aadd Amul : tA -> tA -> tA) := {
distrRight : forall a b c,
Amul (Aadd a b) c = Aadd (Amul a c) (Amul b c)
}.
Get parameter of this structure
Ltac sgroup_head f :=
rewrite ?associative;
repeat
(match goal with
| |- f ?a _ = f ?a _ => apply f_equal2; [reflexivity|]
end;
auto).
Ltac sgroup_tail f :=
rewrite <- ?associative;
repeat
(match goal with
| |- f _ ?a = f _ ?a => apply f_equal2; [|reflexivity]
end;
auto).
Ltac sgroup_only f :=
try (progress (sgroup_head f));
try (progress (sgroup_tail f)).
Ltac sgroup :=
match goal with
| SG : SGroup ?f |- _ => sgroup_only f
end.
Section test.
Context `{HSGroup : SGroup}. Infix "+" := Aadd.
Variable a b c d e f g h i j k : tA.
Goal a + b = a + k. progress sgroup. Abort.
Goal a + b = k + b. progress sgroup. Abort.
Goal a + b + c = a + k. progress sgroup. Abort.
Goal a + b + c = a + b + k. progress sgroup. Abort.
Goal a + b + c = k + c. progress sgroup. Abort.
Goal (a + b) + c = k + (b + c). progress sgroup. Abort.
Goal a + b + c + d = a + k. progress sgroup. Abort.
Goal a + b + c + d = a + b + k. progress sgroup. Abort.
Goal a + b + c + d = a + b + c + k. progress sgroup. Abort.
Goal a + b + c + d = k + d. progress sgroup. Abort.
Goal a + b + c + d = k + c + d. progress sgroup. Abort.
Goal a + b + c + d = k + b + c + d. progress sgroup. Abort.
Goal a + b + c + d + e = a + k. progress sgroup. Abort.
Goal a + b + c + d + e = a + b + k. progress sgroup. Abort.
Goal a + b + c + d + e = a + b + c + k. progress sgroup. Abort.
Goal a + b + c + d + e = a + b + c + d + k. progress sgroup. Abort.
Goal a + b + c + d + e = k + e. progress sgroup. Abort.
Goal a + b + c + d + e = k + d + e. progress sgroup. Abort.
Goal a + b + c + d + e = k + c + d + e. progress sgroup. Abort.
Goal a + b + c + d + e = k + b + c + d + e. progress sgroup. Abort.
End test.
Class ASGroup {tA} (Aadd : tA -> tA -> tA) := {
asgroupSGroup :: SGroup Aadd;
asgroupComm :: Commutative Aadd
}.
asgroupSGroup :: SGroup Aadd;
asgroupComm :: Commutative Aadd
}.
Get parameter of this structure
Extra Theories
In a commutative semigroup, adjust a term in the equation to the tail,
and use full left associative form for next step of elimination.
From: a1 + ... + c + ... + an (with parentheses of any form)
To : (...(a1 + ... + an)...) + c 在交换半群中,将等式里的一个项调整到尾部,并使用完全的左结合形式以便下一步消去。
Ltac move2t c :=
rewrite ?associative;
try rewrite (commutative c);
rewrite <- ?associative.
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 : tA.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof. do 2 move2h a0. do 2 move2h a1. do 2 move2h a2. do 2 move2h a3. auto. Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof. do 2 move2t a0. do 2 move2t a1. do 2 move2t a2. do 2 move2t a3. auto. Qed.
End test.
Section try1.
rewrite ?associative;
try rewrite (commutative c);
rewrite <- ?associative.
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 : tA.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof. do 2 move2h a0. do 2 move2h a1. do 2 move2h a2. do 2 move2h a3. auto. Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof. do 2 move2t a0. do 2 move2t a1. do 2 move2t a2. do 2 move2t a3. auto. Qed.
End test.
Section try1.
In a commutative semigroup, eliminate first common head.
From: c + a1 + ... + an = c + b1 + ... + bm (with parentheses of any form)
To : a1 + (a2 + (... + an)) = b1 + (b2 + (... + bm)) 在交换半群中,消去第一个相同的头部。
In a commutative semigroup, eliminate first common tail.
From: c + a1 + ... + an = c + b1 + ... + bm (with parentheses of any form)
To : ((a1 + a2) + ...) + an = ((b1 + b2) + ...) + bm
在交换半群中,消去第一个相同的尾部。
在交换半群中,自动消去左式中所有可能相同的头部
Ltac elimh :=
rewrite ?associative;
repeat match goal with
| |- ?aeq (?f ?c _) (?f _ _) => move2h c; elimh1
| |- ?aeq (?f ?c _) (?f _ _) => move2h c; elimh1
end.
rewrite ?associative;
repeat match goal with
| |- ?aeq (?f ?c _) (?f _ _) => move2h c; elimh1
| |- ?aeq (?f ?c _) (?f _ _) => move2h c; elimh1
end.
在交换半群中,自动消去左式中所有可能相同的尾部
Ltac elimt :=
rewrite <- ?associative;
repeat match goal with
| |- ?aeq (?f _ ?c) (?f _ _) => move2t c; elimt1
end.
rewrite <- ?associative;
repeat match goal with
| |- ?aeq (?f _ ?c) (?f _ _) => move2t c; elimt1
end.
在交换半群中,自动消去左式和右式中所有可能相同的头部和尾部
Ltac asgroup :=
elimh; elimt;
symmetry;
elimh; elimt.
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 b1 b2 c1 c2 : tA.
elimh; elimt;
symmetry;
elimh; elimt.
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 b1 b2 c1 c2 : tA.
第一种情形,等式两侧完全相同
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof.
do 2 move2h a0; elimh1.
do 2 move2h a1; elimh1.
do 2 move2h a2; elimh1.
Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1.
Proof.
do 2 move2t a0; elimt1.
do 2 move2t a1; elimt1.
do 2 move2t a2; elimt1.
Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1. Proof. elimh. Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1. Proof. elimt. Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1. Proof. symmetry. elimh. Qed.
Goal a0 + (a1 + a2) + a3 = a3 + (a0 + a2) + a1. Proof. symmetry. elimt. Qed.
第二种情形,等式两侧不完全相同
Let eq2 : Prop := a0 + (a1 + a2 + a3) + a4 + a5 = a2 + a0 + a6 + a3 + a4.
Goal eq2. unfold eq2. elimh. Abort.
Goal eq2. unfold eq2. elimt. Abort.
Goal eq2. unfold eq2. symmetry. elimh. Abort.
Goal eq2. unfold eq2. symmetry. elimt. Abort.
Goal eq2. unfold eq2. asgroup. Abort.
Goal eq2. unfold eq2. elimh. Abort.
Goal eq2. unfold eq2. elimt. Abort.
Goal eq2. unfold eq2. symmetry. elimh. Abort.
Goal eq2. unfold eq2. symmetry. elimt. Abort.
Goal eq2. unfold eq2. asgroup. Abort.
第三种情况,某个相同的项出现中中间,既不在头部,也不在尾部
Goal b1 + a0 + a1 + b2 = c1 + a1 + a0 + c2.
Proof.
progress asgroup.
progress asgroup.
progress asgroup.
Abort.
Goal b1 + a0 + a1 + b2 = c1 + a1 + a0 + c2.
Proof.
intros.
do 2 move2h a0. asgroup.
do 2 move2h a1. asgroup.
Abort.
End test.
End try1.
Section try2.
在交换半群中,自动消去等式中相同的项
Ltac asgroup :=
rewrite <- ?associative;
repeat
(match goal with
| |- ?f ?a _ = ?f ?a _ => f_equal
| |- ?f _ ?a = ?f _ ?a => f_equal
| |- ?f ?a _ = ?f _ ?a => move2t a
| |- ?f _ ?a = ?f ?a _ => move2t a
| |- ?f (?f ?a _) _ = ?f ?a _ => move2t a
| |- ?f (?f ?a _) _ = ?f _ ?a => move2t a
| |- ?f (?f _ ?a) _ = ?f ?a _ => move2t a
| |- ?f (?f _ ?a) _ = ?f _ ?a => move2t a
| |- ?f ?a _ = ?f (?f ?a _) _ => move2t a
| |- ?f ?a _ = ?f (?f _ ?a) _ => move2t a
| |- ?f _ ?a = ?f (?f ?a _) _ => move2t a
| |- ?f _ ?a = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f ?a _) _ = ?f (?f ?a _) _ => move2t a
| |- ?f (?f ?a _) _ = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f _ ?a) _ = ?f (?f ?a _) _ => move2t a
| |- ?f (?f _ ?a) _ = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f ?a _) _ = ?f _ (?f ?a _) => move2t a
| |- ?f (?f ?a _) _ = ?f _ (?f _ ?a) => move2t a
| |- ?f (?f _ ?a) _ = ?f _ (?f ?a _) => move2t a
| |- ?f (?f _ ?a) _ = ?f _ (?f _ ?a) => move2t a
end;
auto).
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 b1 b2 c1 c2 : tA.
Goal b1 + a0 + b2 = c1 + a0 + c2.
Proof. asgroup. Abort.
End test.
End try2.
Ltac asgroup_only f :=
rewrite <- ?associative;
repeat
(match goal with
| |- ?A1 = ?A2 =>
match A1 with
| context [f ?a ?b] =>
match A2 with
| context [f a _] => do 2 move2t a; f_equal
| context [f _ a] => do 2 move2t a; f_equal
| context [f b _] => do 2 move2t b; f_equal
| context [f _ b] => do 2 move2t b; f_equal
end
end
end;
auto).
Ltac asgroup :=
match goal with
| ASG : ASGroup ?f |- _ => asgroup_only f
end.
rewrite <- ?associative;
repeat
(match goal with
| |- ?f ?a _ = ?f ?a _ => f_equal
| |- ?f _ ?a = ?f _ ?a => f_equal
| |- ?f ?a _ = ?f _ ?a => move2t a
| |- ?f _ ?a = ?f ?a _ => move2t a
| |- ?f (?f ?a _) _ = ?f ?a _ => move2t a
| |- ?f (?f ?a _) _ = ?f _ ?a => move2t a
| |- ?f (?f _ ?a) _ = ?f ?a _ => move2t a
| |- ?f (?f _ ?a) _ = ?f _ ?a => move2t a
| |- ?f ?a _ = ?f (?f ?a _) _ => move2t a
| |- ?f ?a _ = ?f (?f _ ?a) _ => move2t a
| |- ?f _ ?a = ?f (?f ?a _) _ => move2t a
| |- ?f _ ?a = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f ?a _) _ = ?f (?f ?a _) _ => move2t a
| |- ?f (?f ?a _) _ = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f _ ?a) _ = ?f (?f ?a _) _ => move2t a
| |- ?f (?f _ ?a) _ = ?f (?f _ ?a) _ => move2t a
| |- ?f (?f ?a _) _ = ?f _ (?f ?a _) => move2t a
| |- ?f (?f ?a _) _ = ?f _ (?f _ ?a) => move2t a
| |- ?f (?f _ ?a) _ = ?f _ (?f ?a _) => move2t a
| |- ?f (?f _ ?a) _ = ?f _ (?f _ ?a) => move2t a
end;
auto).
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a0 a1 a2 a3 a4 a5 a6 b1 b2 c1 c2 : tA.
Goal b1 + a0 + b2 = c1 + a0 + c2.
Proof. asgroup. Abort.
End test.
End try2.
Ltac asgroup_only f :=
rewrite <- ?associative;
repeat
(match goal with
| |- ?A1 = ?A2 =>
match A1 with
| context [f ?a ?b] =>
match A2 with
| context [f a _] => do 2 move2t a; f_equal
| context [f _ a] => do 2 move2t a; f_equal
| context [f b _] => do 2 move2t b; f_equal
| context [f _ b] => do 2 move2t b; f_equal
end
end
end;
auto).
Ltac asgroup :=
match goal with
| ASG : ASGroup ?f |- _ => asgroup_only f
end.
Section test.
Context `{HASGroup : ASGroup}. Infix "+" := Aadd.
Variable a b c d e f g h i j k : tA.
Goal a + b = a + k. progress asgroup. Abort.
Goal b + a = k + a. progress asgroup. Abort.
Goal a + b = k + a. progress asgroup. Abort.
Goal b + a = a + k. progress asgroup. Abort.
Goal a + b + c = a + k. progress asgroup. Abort.
Goal a + b + c = k + a. progress asgroup. Abort.
Goal a + b + c = b + k. progress asgroup. Abort.
Goal a + b + c = k + b. progress asgroup. Abort.
Goal a + b + c + d = a + k. progress asgroup. Abort.
Goal a + b + c + d = k + a. progress asgroup. Abort.
Goal a + b + c + d = b + k. progress asgroup. Abort.
Goal a + b + c + d = k + b. progress asgroup. Abort.
Goal a + b + c + d + e = a + k. progress asgroup. Abort.
Goal a + b + c + d + e = k + a. progress asgroup. Abort.
Goal a + b + c + d + e = b + k. progress asgroup. Abort.
Goal a + b + c + d + e = k + b. progress asgroup. Abort.
Goal a + b + c + d + e + f = a + k. progress asgroup. Abort.
Goal a + b + c + d + e + f = k + a. progress asgroup. Abort.
Goal a + b + c + d + e + f = b + k. progress asgroup. Abort.
Goal a + b + c + d + e + f = k + b. progress asgroup. Abort.
Goal a + k = a + b + c + d + e + f. progress asgroup. Abort.
Goal k + a = a + b + c + d + e + f. progress asgroup. Abort.
Goal b + k = a + b + c + d + e + f. progress asgroup. Abort.
Goal k + b = a + b + c + d + e + f. progress asgroup. Abort.
Goal forall x1 x2 y1 y2 : tA, x1 + a + b + c + x2 = y1 + c + (b + a) + y2.
Proof. intros. progress asgroup. Abort.
End test.
Class Monoid {tA} (Aadd : tA -> tA -> tA) (Azero : tA) := {
monoidAssoc :: Associative Aadd;
monoidIdL :: IdentityLeft Aadd Azero;
monoidIdR :: IdentityRight Aadd Azero;
monoidSGroup :: SGroup Aadd
}.
monoidAssoc :: Associative Aadd;
monoidIdL :: IdentityLeft Aadd Azero;
monoidIdR :: IdentityRight Aadd Azero;
monoidSGroup :: SGroup Aadd
}.
Get parameter of a monoid
Section Theory.
Context `{HMonoid : Monoid}.
Notation "0" := Azero : A_scope.
Infix "+" := Aadd : A_scope.
End Theory.
Ltac monoid_only f e :=
repeat
(match goal with
| [M:Monoid f e |- context [f ?a e]] => rewrite (identityRight a) at 1
| [M:Monoid f e, H:context [f ?a e] |- _] => rewrite (identityRight a) in H at 1
| [M:Monoid f e |- context [f e ?a]] => rewrite (identityLeft a) at 1
| [M:Monoid f e, H:context [f e ?a] |- _] => rewrite (identityLeft a) in H at 1
end;
auto).
Ltac monoid :=
try match goal with
| M:Monoid ?f ?e |- _ =>
monoid_only f e;
try pose proof (@monoidSGroup _ f e M) as HSGroup; sgroup
end;
try match goal with
| SG : SGroup ?f |- _ => sgroup
end;
auto.
Section Examples.
Context `{HMonoid : Monoid}.
Notation "0" := Azero : A_scope.
Infix "+" := Aadd : A_scope.
Goal forall a b c d : tA, 0 + a + 0 + (b + 0 + c) = a + (0 + b) + (c + 0).
Proof. intros. monoid. Qed.
End Examples.
Class AMonoid {tA} Aadd Azero := {
amonoidMonoid :: @Monoid tA Aadd Azero;
amonoidComm :: Commutative Aadd;
amonoidASGroup :: ASGroup Aadd
}.
amonoidMonoid :: @Monoid tA Aadd Azero;
amonoidComm :: Commutative Aadd;
amonoidASGroup :: ASGroup Aadd
}.
Section Theory.
Context `(HAMonoid : AMonoid).
Infix "*" := Aadd.
End Theory.
Ltac amonoid :=
repeat
(match goal with
| AM : AMonoid ?f ?e |- _ =>
try pose proof (@amonoidMonoid _ f e AM) as HMonoid; try monoid;
try pose proof (@amonoidASGroup _ f e AM) as HASGroup; asgroup
| [M : Monoid ?f ?e |- _] => monoid
| [ASG : ASGroup ?f |- _] => asgroup
| [SG : ASGroup ?f |- _] => sgroup
end;
auto).
Section Examples.
Context `{HAMonoid : AMonoid}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Goal forall a b c : tA, a + (0 + b + c + 0) = 0 + c + (b + a + 0) + 0.
Proof. logic. amonoid. Qed.
End Examples.
Context `{HAMonoid : AMonoid}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Goal forall a b c : tA, a + (0 + b + c + 0) = 0 + c + (b + a + 0) + 0.
Proof. logic. amonoid. Qed.
End Examples.
Class Group {tA} Aadd Azero (Aopp : tA -> tA) := {
groupMonoid :: Monoid Aadd Azero;
groupInvL :: InverseLeft Aadd Azero Aopp;
groupInvR :: InverseRight Aadd Azero Aopp;
}.
groupMonoid :: Monoid Aadd Azero;
groupInvL :: InverseLeft Aadd Azero Aopp;
groupInvR :: InverseRight Aadd Azero Aopp;
}.
Ltac group_basic_only f e inv :=
repeat
(match goal with
| [G:Group f e inv, H:context[f (inv ?a) ?a] |-_] => rewrite (inverseLeft a) in H at 1
| [G:Group f e inv |- context [f (inv ?a) ?a]] => rewrite (inverseLeft a) at 1
| [G:Group f e inv, H:context[f ?a (inv ?a)] |-_] => rewrite (inverseRight a) in H at 1
| [G:Group f e inv |- context [f ?a (inv ?a)]] => rewrite (inverseRight a) at 1
| [G:Group f e inv, H:context[f (inv ?a) (f ?a ?b)] |- _] =>
rewrite <- (associative (inv a) a) in H; rewrite (inverseLeft a) in H at 1
| [G:Group f e inv |- context[f (inv ?a) (f ?a ?b)] ] =>
rewrite <- (associative (inv a) a); rewrite (inverseLeft a) at 1
| [G:Group f e inv, H:context[f ?a (f (inv ?a) ?b)] |- _] =>
rewrite <- (associative a (inv a)) in H; rewrite (inverseRight a) in H at 1
| [G:Group f e inv |- context[f ?a (f (inv ?a) ?b)] ] =>
rewrite <- (associative a (inv a)); rewrite (inverseRight a) at 1
| [G:Group f e inv, H:context[f (f ?a ?b) (inv ?b)] |- _] =>
rewrite (associative a b) in H; rewrite (inverseRight b) in H at 1
| [G:Group f e inv |- context[f (f ?a ?b) (inv ?b)] ] =>
rewrite (associative a b); rewrite (inverseRight b) at 1
| [G:Group f e inv, H:context[f (f ?a (inv ?b)) ?b] |- _] =>
rewrite (associative a (inv b)) in H; rewrite (inverseLeft b) in H at 1
| [G:Group f e inv |- context[f (f ?a (inv ?b)) ?b] ] =>
rewrite (associative a (inv b)); rewrite (inverseLeft b) at 1
end;
auto).
Ltac group_basic :=
repeat
(match goal with
| G : Group ?f ?e ?inv |- _ =>
group_basic_only f e inv;
try pose proof (@groupMonoid _ f e inv G) as HMonoid;
try pose proof (@monoidSGroup _ f e ?M) as HSGroup;
monoid;
sgroup
| M : Monoid ?f ?e |- _ => monoid
| SG : SGroup ?f |- _ => sgroup
end;
auto).
Section example.
Context `{HGroup : Group}.
Notation "0" := Azero : A_scope.
Infix "+" := Aadd : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Goal forall a b c : tA, a + 0 = 0 + a -> (a + b) + 0 + c = 0 + a + 0 + (b + 0 + c).
Proof. logic. group_basic. Qed.
Goal forall a b c d e : tA, c + -c + - d + d = e -> - a + a + b + - b = e.
Proof. logic. group_basic. Qed.
Goal forall a b : tA, -a + (a + b) = b.
Proof. logic. group_basic. Qed.
End example.
Section GroupTheory.
Context `{HGroup : Group}.
Infix "+" := Aadd.
Notation "0" := Azero.
Notation "- a" := (Aopp a).
Notation Asub a b := (a + -b).
Infix "-" := Asub.
Theorem group_opp_0 : - 0 = 0.
Proof.
pose proof (inverseLeft 0). rewrite identityRight in H. auto.
Qed.
Proof.
pose proof (inverseLeft 0). rewrite identityRight in H. auto.
Qed.
left identity element is unique
Theorem group_id_uniq_l : forall e',
(forall a, e' + a = a) -> e' = 0.
Proof.
intros. rewrite <- H. rewrite identityRight; auto.
Qed.
(forall a, e' + a = a) -> e' = 0.
Proof.
intros. rewrite <- H. rewrite identityRight; auto.
Qed.
right identity element is unique
Theorem group_id_uniq_r : forall e', (forall a, a + e' = a) -> e' = 0.
Proof.
intros. rewrite <- H. rewrite identityLeft; auto.
Qed.
Proof.
intros. rewrite <- H. rewrite identityLeft; auto.
Qed.
left inverse element is unique
Theorem group_opp_uniq_l : forall a b : tA, a + b = 0 -> -a = b.
Proof.
intros.
replace (-a) with (-a + 0) by group_basic.
replace 0 with (a + b). group_basic.
Qed.
Proof.
intros.
replace (-a) with (-a + 0) by group_basic.
replace 0 with (a + b). group_basic.
Qed.
right inverse element is unique
Theorem group_opp_uniq_r : forall a b : tA, a + b = 0 -> -b = a.
Proof.
intros.
replace (-b) with (0 + -b) by group_basic.
replace 0 with (a + b). group_basic.
Qed.
Proof.
intros.
replace (-b) with (0 + -b) by group_basic.
replace 0 with (a + b). group_basic.
Qed.
c + a = c + b -> a = b
Theorem group_cancel_l : forall a b c, c + a = c + b -> a = b.
Proof.
intros.
rewrite <- identityLeft at 1.
assert (0 = -c + c). group_basic.
rewrite H0. rewrite associative. rewrite H. group_basic.
Qed.
Proof.
intros.
rewrite <- identityLeft at 1.
assert (0 = -c + c). group_basic.
rewrite H0. rewrite associative. rewrite H. group_basic.
Qed.
a + c = b + c -> a = b
Theorem group_cancel_r : forall a b c, a + c = b + c -> a = b.
Proof.
intros.
rewrite <- identityRight at 1.
assert (0 = c + (-c)). group_basic.
rewrite H0. rewrite <- associative. rewrite H. group_basic.
Qed.
Proof.
intros.
rewrite <- identityRight at 1.
assert (0 = c + (-c)). group_basic.
rewrite H0. rewrite <- associative. rewrite H. group_basic.
Qed.
- - a = a
Theorem group_opp_opp : forall a, - - a = a.
Proof. intros. apply group_cancel_l with (- a). group_basic. Qed.
Proof. intros. apply group_cancel_l with (- a). group_basic. Qed.
a - a = 0
- a = - b -> a = b
Lemma group_opp_inj : forall a b : tA, - a = - b -> a = b.
Proof.
intros. rewrite <- group_opp_opp. rewrite <- H. rewrite group_opp_opp. auto.
Qed.
Proof.
intros. rewrite <- group_opp_opp. rewrite <- H. rewrite group_opp_opp. auto.
Qed.
- (a + b) = (- b) + (- a)
Theorem group_opp_distr : forall a b, - (a + b) = (- b) + (- a).
Proof.
intros.
apply group_cancel_l with (a + b).
rewrite inverseRight. rewrite <- associative. rewrite (associative a b).
group_basic.
Qed.
Proof.
intros.
apply group_cancel_l with (a + b).
rewrite inverseRight. rewrite <- associative. rewrite (associative a b).
group_basic.
Qed.
- a = 0 <-> a = 0
Lemma group_opp_eq0_iff : forall a : tA, - a = 0 <-> a = 0.
Proof.
intros. split; intros.
- apply group_cancel_l with (-a). group_basic.
- subst. rewrite group_opp_0. auto.
Qed.
Proof.
intros. split; intros.
- apply group_cancel_l with (-a). group_basic.
- subst. rewrite group_opp_0. auto.
Qed.
- a <> 0 <-> a <> 0
Lemma group_opp_neq0_iff : forall a : tA, - a <> 0 <-> a <> 0.
Proof. intros. rewrite group_opp_eq0_iff. easy. Qed.
Proof. intros. rewrite group_opp_eq0_iff. easy. Qed.
a + b = c -> a = c + (-b)
Theorem group_sol_l : forall a b c, a + b = c -> a = c + (- b).
Proof. intros. apply group_cancel_r with (b). rewrite H. group_basic. Qed.
Proof. intros. apply group_cancel_r with (b). rewrite H. group_basic. Qed.
a + b = c -> b = (-a) + c
Theorem group_sol_r : forall a b c, a + b = c -> b = (- a) + c.
Proof.
intros. apply group_cancel_l with (a). rewrite H. group_basic.
Qed.
Proof.
intros. apply group_cancel_l with (a). rewrite H. group_basic.
Qed.
a - b = 0 <-> a = b
Theorem group_sub_eq0_iff_eq : forall a b, a - b = 0 <-> a = b.
Proof.
intros. split; intros.
- apply group_cancel_r with (-b). group_basic.
- subst. group_basic.
Qed.
Proof.
intros. split; intros.
- apply group_cancel_r with (-b). group_basic.
- subst. group_basic.
Qed.
Definition 14.1 (multiple operations)
Definition group_batch (l:list tA) :=
match l with
| [] => 0
| x :: l' => fold_left Aadd l' x
end.
Section test.
Variable (a1 a2 a3 a4 : tA).
End test.
match l with
| [] => 0
| x :: l' => fold_left Aadd l' x
end.
Section test.
Variable (a1 a2 a3 a4 : tA).
End test.
Theorem 14.3 (Generalized Associative Law)
Section th14_3.
Notation "'Σ' a '&' l " := (fold_left Aadd l a) (at level 10).
Theorem group_assoc_general (l1 l2 : list tA) :
(group_batch l1) + (group_batch l2) = group_batch (l1 ++ l2).
Proof.
destruct l1,l2; simpl; group_basic.
- rewrite app_nil_r. auto.
- rename t into a1, t0 into a2.
assert (forall a l1 l2, Σ a & (l1 ++ l2) = Σ (Σ a & l1) & l2) as H1.
{ intros a l0. revert a. induction l0; intros; auto.
simpl. rewrite IHl0. auto. }
assert (forall a b l, a + Σ b & l = Σ (a + b) & l) as H2.
{ intros. revert a b. induction l; simpl; intros; auto.
simpl. rewrite IHl.
Notation "'Σ' a '&' l " := (fold_left Aadd l a) (at level 10).
Theorem group_assoc_general (l1 l2 : list tA) :
(group_batch l1) + (group_batch l2) = group_batch (l1 ++ l2).
Proof.
destruct l1,l2; simpl; group_basic.
- rewrite app_nil_r. auto.
- rename t into a1, t0 into a2.
assert (forall a l1 l2, Σ a & (l1 ++ l2) = Σ (Σ a & l1) & l2) as H1.
{ intros a l0. revert a. induction l0; intros; auto.
simpl. rewrite IHl0. auto. }
assert (forall a b l, a + Σ b & l = Σ (a + b) & l) as H2.
{ intros. revert a b. induction l; simpl; intros; auto.
simpl. rewrite IHl.
fold_left preveres the aeq
assert (forall l a1 a2, a1 = a2 -> Σ a1 & l = Σ a2 & l).
{ induction l0; intros; simpl in *; auto.
apply IHl0. rewrite H. auto. }
apply H. group_basic. }
assert (forall a b l, Σ a & (b :: l) = Σ (a + b) & l) as H3.
{ intros. revert a b. induction l; auto. }
rewrite H1. rewrite H2. rewrite H3. auto.
Qed.
End th14_3.
Section th14_4.
Import ZArith.
{ induction l0; intros; simpl in *; auto.
apply IHl0. rewrite H. auto. }
apply H. group_basic. }
assert (forall a b l, Σ a & (b :: l) = Σ (a + b) & l) as H3.
{ intros. revert a b. induction l; auto. }
rewrite H1. rewrite H2. rewrite H3. auto.
Qed.
End th14_3.
Section th14_4.
Import ZArith.
Definition 14.2 (power)
a ^ 0 = e
a ^ n = a ^ (n-1) * a, for n >= 1
a ^ (-n) = (-a) ^ n, for n >= 1
Definition group_power (a : tA) (n : Z) : tA :=
match n with
| Z0 => 0
| Zpos m => iterate (fun x => Aadd x a) (Pos.to_nat m) 0
| Z.neg m => iterate (fun x => Aadd x (Aopp a)) (Pos.to_nat m) 0
end.
Infix "^" := group_power.
Section test.
Variable (a1 a2 a3 a4 : tA).
End test.
match n with
| Z0 => 0
| Zpos m => iterate (fun x => Aadd x a) (Pos.to_nat m) 0
| Z.neg m => iterate (fun x => Aadd x (Aopp a)) (Pos.to_nat m) 0
end.
Infix "^" := group_power.
Section test.
Variable (a1 a2 a3 a4 : tA).
End test.
Remark 14.2
Lemma group_power_eq1 (n : Z) :
match n with
| Z0 => forall a, a ^ n = 0
| Zpos m => forall a, a ^ n = group_batch (repeat a (Z.to_nat n))
| Zneg m => forall a, a ^ n = group_batch (repeat (-a) (Z.to_nat (-n)))
end.
Proof.
destruct n; intros; auto.
Abort.
match n with
| Z0 => forall a, a ^ n = 0
| Zpos m => forall a, a ^ n = group_batch (repeat a (Z.to_nat n))
| Zneg m => forall a, a ^ n = group_batch (repeat (-a) (Z.to_nat (-n)))
end.
Proof.
destruct n; intros; auto.
Abort.
Theorem 14.4
Theorem group_power_inv : forall a n, (a^n) + (a^(- n)) = 0.
Abort.
Theorem group_power_plus : forall a m n, (a^m) + (a^n) = a^(m+n).
Abort.
Theorem group_power_mul : forall a m n, (a^m)^n = a^(m*n).
Abort.
End th14_4.
Abort.
Theorem group_power_plus : forall a m n, (a^m) + (a^n) = a^(m+n).
Abort.
Theorem group_power_mul : forall a m n, (a^m)^n = a^(m*n).
Abort.
End th14_4.
Lemma group_add_eq_l : forall a b : tA, b = 0 -> a + b = a.
Proof. intros. rewrite H. apply identityRight. Qed.
Lemma group_add_eq_r : forall a b : tA, a = 0 -> a + b = b.
Proof. intros. rewrite H. apply identityLeft. Qed.
Lemma group_add_eq_l_inv : forall a b : tA, a + b = a -> b = 0.
Proof.
intros.
assert (a + b = a + 0). rewrite H. rewrite identityRight; auto.
apply group_cancel_l in H0. auto.
Qed.
Lemma group_add_eq_r_inv : forall a b : tA, a + b = b -> a = 0.
Proof.
intros.
assert (a + b = 0 + b). rewrite H. rewrite identityLeft; auto.
apply group_cancel_r in H0. auto.
Qed.
Lemma group_opp_exchg_l : forall a b : tA, - a = b -> a = - b.
Proof. intros. rewrite <- H. rewrite group_opp_opp. auto. Qed.
Lemma group_opp_exchg_r : forall a b : tA, a = - b -> - a = b.
Proof. intros. rewrite H. rewrite group_opp_opp. auto. Qed.
End GroupTheory.
Ltac group_advanced_only f e inv :=
repeat
(match goal with
| [G:Group f e inv, H:context [inv e] |- _] => rewrite group_opp_0 in H at 1
| [G:Group f e inv |- context [inv e]] => rewrite group_opp_0 at 1
| [G:Group f e inv, H:context [inv (inv ?a)] |- _] =>
rewrite (group_opp_opp a) in H
| [G:Group f e inv |- context [inv (inv ?a)]] =>
rewrite (group_opp_opp a)
| [G:Group f e inv, H:context [inv (f ?a ?b)] |- _] =>
rewrite (group_opp_distr a b) in H at 1
| [G:Group f e inv |- context [inv (f ?a ?b)]] =>
rewrite (group_opp_distr a b) at 1
| [G:Group f e inv, H:context [inv ?a = inv ?b] |- _] =>
apply group_opp_inj in H
| [G:Group f e inv, H:context [inv ?a = ?b] |- ?a = inv ?b] =>
apply group_opp_exchg_l
| [G:Group f e inv, H:context [?a = inv ?b] |- inv ?a = ?b] =>
apply group_opp_exchg_r
| [G:Group f e inv, H:context [inv ?a = e] |- _] => rewrite (group_opp_eq0_iff a) in H
| [G:Group f e inv |- context [inv ?a = e]] => rewrite (group_opp_eq0_iff a)
| [G:Group f e inv |- f ?a ?b = ?a] => apply group_add_eq_l
| [G:Group f e inv |- ?a = f ?a ?b] => symmetry
| [G:Group f e inv |- f ?a ?b = ?b] => apply group_add_eq_r
| [G:Group f e inv |- ?b = f ?a ?b] => symmetry
| [G:Group f e inv, H:f ?a ?b = ?a |- ?b = e] => apply group_add_eq_l_inv in H
| [G:Group f e inv, H:f ?a ?b = ?b |- ?a = e] => apply group_add_eq_r_inv in H
| [G:Group f e inv |- ?f (?f ?a ?b) ?c = ?a] => rewrite (associative a b c)
| [G:Group f e inv |- ?a = ?f (?f ?a ?b) ?c] => symmetry
| [G:Group f e inv |- ?f (?f ?a ?b) ?c = ?b] =>
rewrite (associative a b c); rewrite (commutative b c);
rewrite <- (associative a c b)
| [G:Group f e inv |- ?b = ?f (?f ?a ?b) ?c] => symmetry
| [G:Group f e inv |- ?f ?a (?f ?b ?c) = ?b] =>
rewrite (commutative b c); rewrite <- (associative a c b)
| [G:Group f e inv |- ?b = ?f ?a (?f ?b ?c)] => symmetry
| [G:Group f e inv |- ?f ?a (?f ?b ?c) = ?c] => rewrite <- (associative a b c)
| [G:Group f e inv |- ?c = ?f ?a (?f ?b ?c)] => symmetry
end;
auto).
Ltac group_only f e inv :=
try (progress (group_advanced_only f e inv));
try (progress (group_basic)).
Ltac group :=
match goal with
| G:Group ?f ?e ?inv |- _ => group_only f e inv
end.
Section Examples.
Context `{HGroup : Group}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "-" := Asub : A_scope.
Goal forall a b c d : tA, - (a + 0 + (b - c)) + (- - d + 0) = c - b + - (- d + a).
Proof. intros. group. Qed.
Goal forall a b : tA, b = 0 -> a + b = a.
Proof. intros. group. Qed.
Goal forall a b : tA, a = 0 -> a + b = b.
Proof. intros. group. Qed.
Goal forall a b : tA, a + b = b -> a = 0.
Proof. intros. group. Qed.
Goal forall a b : tA, a + b = a -> b = 0.
Proof. intros. group. Qed.
Goal forall a b : tA, a = - b -> - a = b.
Proof. intros. group. Qed.
Goal forall a b : tA, - a = b -> a = - b.
Proof. intros. group. Qed.
End Examples.
Context `{HGroup : Group}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "-" := Asub : A_scope.
Goal forall a b c d : tA, - (a + 0 + (b - c)) + (- - d + 0) = c - b + - (- d + a).
Proof. intros. group. Qed.
Goal forall a b : tA, b = 0 -> a + b = a.
Proof. intros. group. Qed.
Goal forall a b : tA, a = 0 -> a + b = b.
Proof. intros. group. Qed.
Goal forall a b : tA, a + b = b -> a = 0.
Proof. intros. group. Qed.
Goal forall a b : tA, a + b = a -> b = 0.
Proof. intros. group. Qed.
Goal forall a b : tA, a = - b -> - a = b.
Proof. intros. group. Qed.
Goal forall a b : tA, - a = b -> a = - b.
Proof. intros. group. Qed.
End Examples.
Class AGroup {tA} Aadd (Azero : tA) Aopp := {
agroupGroup :: Group Aadd Azero Aopp;
agroupAM :: AMonoid Aadd Azero;
agroupComm :: Commutative Aadd;
}.
agroupGroup :: Group Aadd Azero Aopp;
agroupAM :: AMonoid Aadd Azero;
agroupComm :: Commutative Aadd;
}.
Section Theory.
Context `{HAGroup : AGroup}.
Infix "+" := Aadd.
Notation "- a" := (Aopp a).
Notation "a - b" := (a + (-b)).
Context `{HAGroup : AGroup}.
Infix "+" := Aadd.
Notation "- a" := (Aopp a).
Notation "a - b" := (a + (-b)).
a - b = - (b - a)
Lemma agroup_sub_comm : forall a b, a - b = - (b - a).
Proof.
intros. rewrite group_opp_distr. rewrite group_opp_opp. easy.
Qed.
Proof.
intros. rewrite group_opp_distr. rewrite group_opp_opp. easy.
Qed.
- (a + b) = -a + (-b)
Lemma agroup_sub_distr : forall a b, - (a + b) = -a + (-b).
Proof.
intros. rewrite group_opp_distr. apply commutative.
Qed.
Proof.
intros. rewrite group_opp_distr. apply commutative.
Qed.
(a - b) - c = (a - c) - b
Lemma agroup_sub_perm : forall a b c, (a - b) - c = (a - c) - b.
Proof.
intros. rewrite ?associative. rewrite (commutative (-b)). easy.
Qed.
Proof.
intros. rewrite ?associative. rewrite (commutative (-b)). easy.
Qed.
(a - b) - c = a - (b + c)
Lemma agroup_sub_assoc : forall a b c, (a - b) - c = a - (b + c).
Proof.
intros. rewrite ?associative. rewrite agroup_sub_distr. easy.
Qed.
End Theory.
Ltac agroup_only f e inv :=
repeat
(match goal with
| AG : AGroup f e inv |- context[f (f ?a ?b) (inv ?a)] =>
rewrite (associative a b (inv a));
rewrite (commutative b (inv a));
rewrite <- (associative a (inv a) b)
end;
auto).
Ltac agroup :=
try match goal with
| AG : AGroup ?f ?e ?inv |- _ =>
try pose proof (@agroupGroup _ f e inv AG) as HGroup;
try pose proof (@agroupAM _ f e inv AG) as HAMonoid;
repeat
(agroup_only f e inv;
group_only f e inv;
amonoid)
end;
try match goal with
| G : Group ?f ?e ?inv |- _ =>
try pose proof (@agroupGroup _ f e inv G) as HGroup;
group_only f e inv
end;
try match goal with
| AM : AMonoid ?f ?e |- _ => amonoid
end;
try match goal with
| M : Monoid ?f ?e |- _ => monoid
end;
try match goal with
| ASG : ASGroup ?f |- _ => asgroup
end;
try match goal with
| SG : SGroup ?f |- _ => sgroup
end.
Proof.
intros. rewrite ?associative. rewrite agroup_sub_distr. easy.
Qed.
End Theory.
Ltac agroup_only f e inv :=
repeat
(match goal with
| AG : AGroup f e inv |- context[f (f ?a ?b) (inv ?a)] =>
rewrite (associative a b (inv a));
rewrite (commutative b (inv a));
rewrite <- (associative a (inv a) b)
end;
auto).
Ltac agroup :=
try match goal with
| AG : AGroup ?f ?e ?inv |- _ =>
try pose proof (@agroupGroup _ f e inv AG) as HGroup;
try pose proof (@agroupAM _ f e inv AG) as HAMonoid;
repeat
(agroup_only f e inv;
group_only f e inv;
amonoid)
end;
try match goal with
| G : Group ?f ?e ?inv |- _ =>
try pose proof (@agroupGroup _ f e inv G) as HGroup;
group_only f e inv
end;
try match goal with
| AM : AMonoid ?f ?e |- _ => amonoid
end;
try match goal with
| M : Monoid ?f ?e |- _ => monoid
end;
try match goal with
| ASG : ASGroup ?f |- _ => asgroup
end;
try match goal with
| SG : SGroup ?f |- _ => sgroup
end.
Section Examples.
Context `{HAGroup : AGroup}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "-" := Asub : A_scope.
Goal forall a b c d : tA, - (a + 0 + (b - c)) + (- - d + 0) = c - b + - (- d + a).
Proof. intros. agroup. Qed.
Goal forall a b c : tA, ((a - b) - c = a - (b + c)).
Proof. intros. agroup. Qed.
End Examples.
Context `{HAGroup : AGroup}.
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "-" := Asub : A_scope.
Goal forall a b c d : tA, - (a + 0 + (b - c)) + (- - d + 0) = c - b + - (- d + a).
Proof. intros. agroup. Qed.
Goal forall a b c : tA, ((a - b) - c = a - (b + c)).
Proof. intros. agroup. Qed.
End Examples.
Class SRing {tA} Aadd (Azero : tA) Amul Aone := {
sringAddAM :: AMonoid Aadd Azero;
sringMulM :: Monoid Amul Aone;
sringMul0L : forall a, Amul Azero a = Azero;
sringMul0R : forall a, Amul a Azero = Azero;
sringDistrL :: DistrLeft Aadd Amul;
sringDistrR :: DistrRight Aadd Amul;
}.
Section Theory.
Context `{HSRing : SRing}.
Infix "+" := Aadd.
Infix "*" := Amul.
End Theory.
Ltac sring_only add zero mul one :=
try match goal with
| [HSRing:SRing add zero mul one, H:context[mul zero ?a] |- _] =>
rewrite sringMul0L in H at 1
| [HSRing:SRing add zero mul one, H:context[mul ?a zero] |- _] =>
rewrite sringMul0R in H at 1
| [HSRing:SRing add zero mul one |- context[mul zero ?a]] =>
rewrite sringMul0L at 1
| [HSRing:SRing add zero mul one |- context[mul ?a zero]] =>
rewrite sringMul0R at 1
end;
try rewrite ?distrLeft, ?distrRight in *.
Ltac sring :=
try match goal with
| HSRing : SRing ?add ?zero ?mul ?one |- _ =>
sring_only add zero mul one;
try pose proof (@sringAddAM _ add zero mul one HSRing) as HAddAM;
try pose proof (@sringMulM _ add zero mul one HSRing) as HMulM;
repeat
(amonoid)
end;
try match goal with
| AM : AMonoid ?f ?e |- _ => amonoid
end;
try match goal with
| M : Monoid ?f ?e |- _ => monoid
end;
try match goal with
| ASG : ASGroup ?f |- _ => asgroup
end;
try match goal with
| SG : SGroup ?f |- _ => sgroup
end.
Context `{HSRing : SRing}.
Infix "+" := Aadd.
Infix "*" := Amul.
End Theory.
Ltac sring_only add zero mul one :=
try match goal with
| [HSRing:SRing add zero mul one, H:context[mul zero ?a] |- _] =>
rewrite sringMul0L in H at 1
| [HSRing:SRing add zero mul one, H:context[mul ?a zero] |- _] =>
rewrite sringMul0R in H at 1
| [HSRing:SRing add zero mul one |- context[mul zero ?a]] =>
rewrite sringMul0L at 1
| [HSRing:SRing add zero mul one |- context[mul ?a zero]] =>
rewrite sringMul0R at 1
end;
try rewrite ?distrLeft, ?distrRight in *.
Ltac sring :=
try match goal with
| HSRing : SRing ?add ?zero ?mul ?one |- _ =>
sring_only add zero mul one;
try pose proof (@sringAddAM _ add zero mul one HSRing) as HAddAM;
try pose proof (@sringMulM _ add zero mul one HSRing) as HMulM;
repeat
(amonoid)
end;
try match goal with
| AM : AMonoid ?f ?e |- _ => amonoid
end;
try match goal with
| M : Monoid ?f ?e |- _ => monoid
end;
try match goal with
| ASG : ASGroup ?f |- _ => asgroup
end;
try match goal with
| SG : SGroup ?f |- _ => sgroup
end.
Section Examples.
Context `{HSRing : SRing}.
Infix "+" := Aadd.
Infix "*" := Amul.
Notation "0" := Azero.
Goal forall a b, a * 0 = b -> b = 0.
Proof. intros. sring. Qed.
Goal forall a b, 0 * a = b -> b = 0.
Proof. intros. sring. Qed.
Goal forall a, 0 * a = 0.
Proof. intros. sring. Qed.
Goal forall a, a * 0 = 0.
Proof. intros. sring. Qed.
End Examples.
Class Ring {tA} Aadd (Azero : tA) Aopp Amul Aone := {
ringAddAG :: AGroup Aadd Azero Aopp;
ringSRing :: SRing Aadd Azero Amul Aone;
ringMulM :: Monoid Amul Aone;
ringDistrL :: DistrLeft Aadd Amul;
ringDistrR :: DistrRight Aadd Amul;
}.
Section Theory.
Context `{HRing : Ring}.
Infix "+" := Aadd : A_scope.
Notation "1" := Aone : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "*" := Amul : A_scope.
Context `{HRing : Ring}.
Infix "+" := Aadd : A_scope.
Notation "1" := Aone : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + -b).
Infix "*" := Amul : A_scope.
a + a = (1 + 1) * a
Lemma ring_add_same_eq_mul2 : forall a, a + a = (1 + 1) * a.
Proof. intros. rewrite distrRight, !identityLeft at 1. auto. Qed.
End Theory.
Proof. intros. rewrite distrRight, !identityLeft at 1. auto. Qed.
End Theory.
Class ARing {tA} Aadd Azero Aopp Amul Aone := {
aringRing :: @Ring tA Aadd Azero Aopp Amul Aone;
aringMulComm :: Commutative Amul;
aringMulAMonoid :: AMonoid Amul Aone;
aringMulASGroup :: ASGroup Amul
}.
Lemma make_ring_theory `(HARing : ARing)
: ring_theory Azero Aone Aadd Amul (fun x y => Aadd x (Aopp y)) Aopp eq.
Proof.
constructor; intros;
try (rewrite ?identityLeft,?associative; reflexivity);
try (rewrite commutative; reflexivity).
rewrite distrRight; reflexivity.
rewrite inverseRight; reflexivity.
Qed.
Section Theory.
Context `{HARing : ARing}.
Add Ring ring_inst : (make_ring_theory HARing).
Infix "+" := Aadd.
Notation "0" := Azero.
Notation "- a" := (Aopp a).
Notation Asub a b := (a + -b).
Infix "*" := Amul.
a * 0 = 0
0 * a = 0
a * a = 1, then a = 1 or a = -1
Lemma ring_sqr_eq1_imply_1_neg1 : forall (a : tA),
a * a = Aone -> a = Aone \/ a = (- Aone).
Proof.
Admitted.
a * a = Aone -> a = Aone \/ a = (- Aone).
Proof.
Admitted.
(- a) * b = - (a * b)
Theorem ring_mul_opp_l : forall a b, (- a) * b = - (a * b).
Proof.
intros.
assert (a * b + (-a) * b = 0).
- rewrite <- distrRight. rewrite inverseRight. apply ring_mul_0_l.
- symmetry. apply group_opp_uniq_l. auto.
Qed.
Proof.
intros.
assert (a * b + (-a) * b = 0).
- rewrite <- distrRight. rewrite inverseRight. apply ring_mul_0_l.
- symmetry. apply group_opp_uniq_l. auto.
Qed.
a * (- b) = - (a * b)
Theorem ring_mul_opp_r : forall a b, a * (- b) = - (a * b).
Proof.
intros. rewrite commutative. rewrite ring_mul_opp_l. rewrite commutative; auto.
Qed.
Proof.
intros. rewrite commutative. rewrite ring_mul_opp_l. rewrite commutative; auto.
Qed.
(- a) * (- b) = a * b
Theorem ring_mul_opp_opp : forall a b, (- a) * (- b) = a * b.
Proof.
intros. rewrite ring_mul_opp_l, ring_mul_opp_r. apply group_opp_opp.
Qed.
End Theory.
Proof.
intros. rewrite ring_mul_opp_l, ring_mul_opp_r. apply group_opp_opp.
Qed.
End Theory.
Examples
Module Demo_AbsARing.
Context `{HARing : ARing}.
Infix "+" := Aadd. Infix "*" := Amul.
Notation "0" := Azero. Notation "1" := Aone.
Add Ring ring_inst : (make_ring_theory HARing).
Goal forall a b c : tA, (a + b) * c = 0 + b * c * 1 + 0 + 1 * c * a.
Proof. intros. ring. Qed.
End Demo_AbsARing.
Context `{HARing : ARing}.
Infix "+" := Aadd. Infix "*" := Amul.
Notation "0" := Azero. Notation "1" := Aone.
Add Ring ring_inst : (make_ring_theory HARing).
Goal forall a b c : tA, (a + b) * c = 0 + b * c * 1 + 0 + 1 * c * a.
Proof. intros. ring. Qed.
End Demo_AbsARing.
This is a concrete ring structure
Module Demo_ConcrateRing.
Inductive tA := Azero | A1 | A2 | A3.
Notation "0" := Azero. Notation "1" := A1.
Notation "2" := A2. Notation "3" := A3.
Definition add (a b : tA) :=
match a,b with
| 0,_ => b
| 1,0 => 1 | 1,1 => 2 | 1,2 => 3 | 1,3 => 0
| 2,0 => 2 | 2,1 => 3 | 2,2 => 0 | 2,3 => 1
| 3,0 => 3 | 3,1 => 0 | 3,2 => 1 | 3,3 => 2
end.
Infix "+" := add.
Definition opp (a : tA) :=
match a with
| 0 => 0 | 1 => 3 | 2 => 2 | 3 => 1
end.
Notation "- a" := (opp a).
Notation "a - b" := (a + (-b)).
Definition mul (a b : tA) :=
match a,b with
| 0,_ => 0
| 1,_ => b
| 2,0 => 0 | 2,1 => 2 | 2,2 => 0 | 2,3 => 2
| 3,0 => 0 | 3,1 => 3 | 3,2 => 2 | 3,3 => 1
end.
Infix "*" := mul.
Lemma add_comm : forall a b, a + b = b + a.
Proof. destruct a,b; auto. Qed.
Lemma ring_thy : ring_theory 0 1 add mul (fun x y => add x (opp y)) opp eq.
Proof.
constructor; auto; intros;
destruct x; auto; destruct y; auto; destruct z; auto.
Qed.
Add Ring ring_thy_inst1 : ring_thy.
Local Instance ARing_inst : ARing add 0 opp mul 1.
Proof.
repeat constructor; intros;
destruct a; auto; destruct b; auto; destruct c; auto.
Qed.
Goal forall a b c : tA, a + b + c - b = a + c.
Proof.
intros. ring.
Qed.
End Demo_ConcrateRing.
Inductive tA := Azero | A1 | A2 | A3.
Notation "0" := Azero. Notation "1" := A1.
Notation "2" := A2. Notation "3" := A3.
Definition add (a b : tA) :=
match a,b with
| 0,_ => b
| 1,0 => 1 | 1,1 => 2 | 1,2 => 3 | 1,3 => 0
| 2,0 => 2 | 2,1 => 3 | 2,2 => 0 | 2,3 => 1
| 3,0 => 3 | 3,1 => 0 | 3,2 => 1 | 3,3 => 2
end.
Infix "+" := add.
Definition opp (a : tA) :=
match a with
| 0 => 0 | 1 => 3 | 2 => 2 | 3 => 1
end.
Notation "- a" := (opp a).
Notation "a - b" := (a + (-b)).
Definition mul (a b : tA) :=
match a,b with
| 0,_ => 0
| 1,_ => b
| 2,0 => 0 | 2,1 => 2 | 2,2 => 0 | 2,3 => 2
| 3,0 => 0 | 3,1 => 3 | 3,2 => 2 | 3,3 => 1
end.
Infix "*" := mul.
Lemma add_comm : forall a b, a + b = b + a.
Proof. destruct a,b; auto. Qed.
Lemma ring_thy : ring_theory 0 1 add mul (fun x y => add x (opp y)) opp eq.
Proof.
constructor; auto; intros;
destruct x; auto; destruct y; auto; destruct z; auto.
Qed.
Add Ring ring_thy_inst1 : ring_thy.
Local Instance ARing_inst : ARing add 0 opp mul 1.
Proof.
repeat constructor; intros;
destruct a; auto; destruct b; auto; destruct c; auto.
Qed.
Goal forall a b c : tA, a + b + c - b = a + c.
Proof.
intros. ring.
Qed.
End Demo_ConcrateRing.
Class Field {tA} Aadd (Azero : tA) Aopp Amul Aone Ainv := {
fieldRing :: ARing Aadd Azero Aopp Amul Aone;
field_mulInvL : forall a, a <> Azero -> Amul (Ainv a) a = Aone;
field_1_neq_0 : Aone <> Azero;
}.
fieldRing :: ARing Aadd Azero Aopp Amul Aone;
field_mulInvL : forall a, a <> Azero -> Amul (Ainv a) a = Aone;
field_1_neq_0 : Aone <> Azero;
}.
Lemma make_field_theory `(HField : Field)
: field_theory Azero Aone Aadd Amul
(fun a b => Aadd a (Aopp b)) Aopp
(fun a b => Amul a (Ainv b)) Ainv eq.
Proof.
constructor; intros;
try (rewrite ?identityLeft,?associative; reflexivity);
try (rewrite commutative; reflexivity).
apply (make_ring_theory fieldRing).
apply field_1_neq_0.
apply field_mulInvL. auto.
Qed.
Lemma make_field_agroup_add `(HField : Field) : AGroup Aadd Azero Aopp.
Proof. apply HField. Qed.
Lemma make_field_amonoid_add `(HField : Field) : AMonoid Aadd Azero.
Proof. apply HField. Qed.
Section Theory.
Context `{HField : Field}.
Infix "+" := Aadd.
Notation "- a" := (Aopp a).
Notation Asub a b := (a + -b).
Infix "-" := Asub.
Notation "0" := Azero.
Notation "1" := Aone.
Infix "*" := Amul.
Notation "/ a" := (Ainv a).
Notation Adiv a b := (a * (/b)).
Infix "/" := Adiv.
Add Field field_inst : (make_field_theory HField).
a <> 0 -> a * / a = 1
Lemma field_mulInvR : forall a : tA, a <> 0 -> a * /a = 1.
Proof. intros. rewrite commutative. rewrite field_mulInvL; easy. Qed.
Proof. intros. rewrite commutative. rewrite field_mulInvL; easy. Qed.
a <> 0 -> (1/a) * a = 1
Lemma field_mulInvL_inv1 : forall a : tA, a <> 0 -> (1/a) * a = 1.
Proof. intros. simpl. field; auto. Qed.
Proof. intros. simpl. field; auto. Qed.
a <> 0 -> a * (1/a) = 1
Lemma field_mulInvR_inv1 : forall a : tA, a <> 0 -> a * (1/a) = 1.
Proof. intros. simpl. field. auto. Qed.
Proof. intros. simpl. field. auto. Qed.
a <> 0 -> / a <> 0
Lemma field_inv_neq0_if_neq0 : forall a : tA, a <> 0 -> / a <> 0.
Proof.
intros. intro.
pose proof (field_mulInvL H). rewrite H0 in H1. rewrite ring_mul_0_l in H1.
apply field_1_neq_0. auto.
Qed.
Proof.
intros. intro.
pose proof (field_mulInvL H). rewrite H0 in H1. rewrite ring_mul_0_l in H1.
apply field_1_neq_0. auto.
Qed.
- 1 <> 0
Lemma field_neg1_neq_0 : - (1) <> 0.
Proof.
intro. rewrite <- group_opp_0 in H at 1. apply group_opp_inj in H.
apply field_1_neq_0; auto.
Qed.
Proof.
intro. rewrite <- group_opp_0 in H at 1. apply group_opp_inj in H.
apply field_1_neq_0; auto.
Qed.
a <> 0 -> a * b = a * c -> b = c
Lemma field_mul_cancel_l : forall a b c : tA,
a <> 0 -> a * b = a * c -> b = c.
Proof.
intros.
assert (/a * (a * b) = /a * (a * c)).
{ rewrite H0. easy. }
rewrite <- ?associative in H1.
rewrite field_mulInvL in H1; auto.
rewrite ?identityLeft in H1. easy.
Qed.
a <> 0 -> a * b = a * c -> b = c.
Proof.
intros.
assert (/a * (a * b) = /a * (a * c)).
{ rewrite H0. easy. }
rewrite <- ?associative in H1.
rewrite field_mulInvL in H1; auto.
rewrite ?identityLeft in H1. easy.
Qed.
c <> 0 -> a * c = b * c -> a = b
Lemma field_mul_cancel_r : forall a b c : tA,
c <> 0 -> a * c = b * c -> a = b.
Proof.
intros.
assert ((a * c) * /c = (b * c) * /c).
{ rewrite H0. easy. }
rewrite ?associative in H1.
rewrite field_mulInvR in H1; auto.
rewrite ?identityRight in H1. easy.
Qed.
c <> 0 -> a * c = b * c -> a = b.
Proof.
intros.
assert ((a * c) * /c = (b * c) * /c).
{ rewrite H0. easy. }
rewrite ?associative in H1.
rewrite field_mulInvR in H1; auto.
rewrite ?identityRight in H1. easy.
Qed.
a * b = 1 -> / a = b
Lemma field_inv_eq_l : forall a b : tA, a <> 0 -> a * b = 1 -> / a = b.
Proof.
intros. apply (@field_mul_cancel_l a (/ a) b); auto.
rewrite field_mulInvR; auto.
Qed.
Proof.
intros. apply (@field_mul_cancel_l a (/ a) b); auto.
rewrite field_mulInvR; auto.
Qed.
a * b = 1 -> / b = a
Lemma field_inv_eq_r : forall a b : tA, b <> 0 -> a * b = 1 -> / b = a.
Proof.
intros. apply (@field_mul_cancel_r (/ b) a b); auto.
rewrite field_mulInvL; auto.
Qed.
Proof.
intros. apply (@field_mul_cancel_r (/ b) a b); auto.
rewrite field_mulInvL; auto.
Qed.
/ / a = a
Lemma field_inv_inv : forall a : tA, a <> 0 -> / / a = a.
Proof.
intros. pose proof (field_inv_neq0_if_neq0 H).
apply field_mul_cancel_l with (/ a); auto.
rewrite field_mulInvL; auto. rewrite field_mulInvR; auto.
Qed.
Proof.
intros. pose proof (field_inv_neq0_if_neq0 H).
apply field_mul_cancel_l with (/ a); auto.
rewrite field_mulInvL; auto. rewrite field_mulInvR; auto.
Qed.
/ a = / b -> a = b
Lemma field_inv_inj : forall a b : tA, a <> 0 -> b <> 0 -> / a = / b -> a = b.
Proof.
intros. rewrite <- field_inv_inv; auto. rewrite <- H1.
rewrite field_inv_inv; auto.
Qed.
Proof.
intros. rewrite <- field_inv_inv; auto. rewrite <- H1.
rewrite field_inv_inv; auto.
Qed.
/ (- a) = - (/ a)
Lemma field_inv_opp : forall a : tA, a <> 0 -> / (- a) = - (/ a).
Proof.
intros. apply field_inv_eq_l. apply group_opp_neq0_iff; auto.
rewrite ring_mul_opp_opp. apply field_mulInvR; auto.
Qed.
Proof.
intros. apply field_inv_eq_l. apply group_opp_neq0_iff; auto.
rewrite ring_mul_opp_opp. apply field_mulInvR; auto.
Qed.
a - b = b - a -> a = b
Lemma field_sub_anti_eq_imply_eq : forall a b, a - b = b - a -> a = b.
Proof.
intros.
assert ((a - b) + b = (b - a) + b). rewrite H. auto.
rewrite associative in H0. rewrite inverseLeft, identityRight in H0.
rewrite commutative in H0. rewrite <- associative in H0.
assert (a + a = b + b - a + a). rewrite H0 at 1. auto.
rewrite associative in H1. rewrite inverseLeft, identityRight in H1 at 1.
replace (a + a) with ((1 + 1) * a) in H1
by (rewrite <- ring_add_same_eq_mul2; auto).
replace (b + b) with ((1 + 1) * b) in H1
by (rewrite <- ring_add_same_eq_mul2; auto).
apply field_mul_cancel_l in H1; auto.
Abort.
Proof.
intros.
assert ((a - b) + b = (b - a) + b). rewrite H. auto.
rewrite associative in H0. rewrite inverseLeft, identityRight in H0.
rewrite commutative in H0. rewrite <- associative in H0.
assert (a + a = b + b - a + a). rewrite H0 at 1. auto.
rewrite associative in H1. rewrite inverseLeft, identityRight in H1 at 1.
replace (a + a) with ((1 + 1) * a) in H1
by (rewrite <- ring_add_same_eq_mul2; auto).
replace (b + b) with ((1 + 1) * b) in H1
by (rewrite <- ring_add_same_eq_mul2; auto).
apply field_mul_cancel_l in H1; auto.
Abort.
a = - a -> a = 0
Lemma field_eq_opp_imply_eq0 : forall a, a = -a -> a = 0.
Proof.
intros.
assert (a + a = 0 + 0). rewrite H at 1. field.
replace (a + a) with ((1 + 1) * a) in H0
by (rewrite <- ring_add_same_eq_mul2; auto).
replace (0 + 0) with ((1 + 1) * 0) in H0
by (rewrite <- ring_add_same_eq_mul2; auto).
apply field_mul_cancel_l in H0; auto.
Abort.
Section Dec.
Context {AeqDec : Dec (@eq tA)}.
Proof.
intros.
assert (a + a = 0 + 0). rewrite H at 1. field.
replace (a + a) with ((1 + 1) * a) in H0
by (rewrite <- ring_add_same_eq_mul2; auto).
replace (0 + 0) with ((1 + 1) * 0) in H0
by (rewrite <- ring_add_same_eq_mul2; auto).
apply field_mul_cancel_l in H0; auto.
Abort.
Section Dec.
Context {AeqDec : Dec (@eq tA)}.
a * b = 0 <-> a = 0 \/ b = 0
Lemma field_mul_eq0_iff : forall a b : tA, a * b = 0 <-> a = 0 \/ b = 0.
Proof.
intros. split; intros.
- destruct (Aeqdec a 0), (Aeqdec b 0); auto.
assert ((/a * a) * (b * /b) = /a * (a * b) * /b). field; auto.
rewrite H, field_mulInvL, field_mulInvR, identityLeft in H0; auto.
rewrite ring_mul_0_r in H0 at 1.
rewrite ring_mul_0_l in H0 at 1.
exfalso. apply field_1_neq_0; auto.
- destruct H; subst.
rewrite ring_mul_0_l; auto. rewrite ring_mul_0_r; auto.
Qed.
Proof.
intros. split; intros.
- destruct (Aeqdec a 0), (Aeqdec b 0); auto.
assert ((/a * a) * (b * /b) = /a * (a * b) * /b). field; auto.
rewrite H, field_mulInvL, field_mulInvR, identityLeft in H0; auto.
rewrite ring_mul_0_r in H0 at 1.
rewrite ring_mul_0_l in H0 at 1.
exfalso. apply field_1_neq_0; auto.
- destruct H; subst.
rewrite ring_mul_0_l; auto. rewrite ring_mul_0_r; auto.
Qed.
a * b <> 0 <-> (a <> 0 /\ b <> 0)
Lemma field_mul_neq0_iff : forall a b : tA, a * b <> 0 <-> (a <> 0 /\ b <> 0).
Proof. intros. rewrite field_mul_eq0_iff. tauto. Qed.
Proof. intros. rewrite field_mul_eq0_iff. tauto. Qed.
/ a <> 0 -> a <> 0
Lemma field_inv_neq0_imply_neq0 : forall a : tA, / a <> 0 -> a <> 0.
Proof.
intros. intro.
Admitted.
Proof.
intros. intro.
Admitted.
/ a <> 0 <-> a <> 0
Lemma field_inv_neq0_iff : forall a : tA, / a <> 0 <-> a <> 0.
Proof.
intros; split. apply field_inv_neq0_imply_neq0.
apply field_inv_neq0_if_neq0.
Qed.
Proof.
intros; split. apply field_inv_neq0_imply_neq0.
apply field_inv_neq0_if_neq0.
Qed.
a * a = 0 -> a = 0
Lemma field_sqr_eq0_reg : forall a : tA, a * a = 0 -> a = 0.
Proof. intros. apply field_mul_eq0_iff in H. destruct H; auto. Qed.
Proof. intros. apply field_mul_eq0_iff in H. destruct H; auto. Qed.
a * b = b -> a = 1 \/ b = 0
Lemma field_mul_eq_imply_a1_or_b0 : forall (a b : tA),
a * b = b -> (a = 1) \/ (b = 0).
Proof.
intros. destruct (Aeqdec a 1), (Aeqdec b 0); auto.
replace b with (1 * b) in H at 2 by ring.
apply field_mul_cancel_r in H; auto.
Qed.
End Dec.
End Theory.
a * b = b -> (a = 1) \/ (b = 0).
Proof.
intros. destruct (Aeqdec a 1), (Aeqdec b 0); auto.
replace b with (1 * b) in H at 2 by ring.
apply field_mul_cancel_r in H; auto.
Qed.
End Dec.
End Theory.
Class Order {tA} (Alt Ale : tA -> tA -> Prop) := {
lt_irrefl : forall a : tA, ~ (Alt a a);
lt_antisym : forall a b : tA, Alt a b -> Alt b a -> a = b;
lt_trans : forall a b c : tA, Alt a b -> Alt b c -> Alt a c;
lt_cases : forall a b : tA, {Alt a b} + {Alt b a} + {a = b};
lt_le_cong : forall a b : tA, Ale a b <-> (Alt a b \/ a = b);
}.
lt_irrefl : forall a : tA, ~ (Alt a a);
lt_antisym : forall a b : tA, Alt a b -> Alt b a -> a = b;
lt_trans : forall a b c : tA, Alt a b -> Alt b c -> Alt a c;
lt_cases : forall a b : tA, {Alt a b} + {Alt b a} + {a = b};
lt_le_cong : forall a b : tA, Ale a b <-> (Alt a b \/ a = b);
}.
eq is decidable
Lemma Aeq_dec : forall a b : tA, {a = b} + {a <> b}.
Proof.
intros; destruct (lt_cases a b) as [[H1|H1]|H1]; auto.
- right; intro; subst. apply lt_irrefl in H1; auto.
- right; intro; subst. apply lt_irrefl in H1; auto.
Qed.
#[export] Instance Aeq_Dec : Dec (@eq tA).
Proof. constructor. intros. apply Aeq_dec. Qed.
End eq.
Proof.
intros; destruct (lt_cases a b) as [[H1|H1]|H1]; auto.
- right; intro; subst. apply lt_irrefl in H1; auto.
- right; intro; subst. apply lt_irrefl in H1; auto.
Qed.
#[export] Instance Aeq_Dec : Dec (@eq tA).
Proof. constructor. intros. apply Aeq_dec. Qed.
End eq.
About "lt"
Section lt.
Context `{HOrder : Order}.
Notation "a > b" := (Alt b a).
Infix "<" := Alt.
Lemma Alt_dec : forall a b, {a < b} + {~(a < b)}.
Proof.
intros. destruct (lt_cases a b) as [[H1|H1]|H1]; auto.
- right. intro. pose proof (lt_trans H1 H). apply lt_irrefl in H0; auto.
- right. subst. apply lt_irrefl.
Qed.
#[export] Instance Alt_Dec : Dec Alt.
Proof. constructor. intros. apply Alt_dec. Defined.
Lemma lt_connected : forall a b : tA, Alt a b \/ Alt b a \/ a = b.
Proof. intros. destruct (lt_cases a b) as [[H1|H1]|H1]; auto. Qed.
Context `{HOrder : Order}.
Notation "a > b" := (Alt b a).
Infix "<" := Alt.
Lemma Alt_dec : forall a b, {a < b} + {~(a < b)}.
Proof.
intros. destruct (lt_cases a b) as [[H1|H1]|H1]; auto.
- right. intro. pose proof (lt_trans H1 H). apply lt_irrefl in H0; auto.
- right. subst. apply lt_irrefl.
Qed.
#[export] Instance Alt_Dec : Dec Alt.
Proof. constructor. intros. apply Alt_dec. Defined.
Lemma lt_connected : forall a b : tA, Alt a b \/ Alt b a \/ a = b.
Proof. intros. destruct (lt_cases a b) as [[H1|H1]|H1]; auto. Qed.
a < b -> a <> b
Lemma lt_not_eq : forall a b : tA, a < b -> a <> b.
Proof. intros. intro. subst. apply lt_irrefl in H; auto. Qed.
Proof. intros. intro. subst. apply lt_irrefl in H; auto. Qed.
a < b -> b < a -> False
Lemma lt_gt_contr : forall a b : tA, a < b -> b < a -> False.
Proof.
intros. apply lt_trans with (a:=a) in H0; auto. apply lt_irrefl in H0; auto.
Qed.
Proof.
intros. apply lt_trans with (a:=a) in H0; auto. apply lt_irrefl in H0; auto.
Qed.
a < b -> a <> b
Lemma lt_imply_neq : forall a b : tA, a < b -> a <> b.
Proof. intros. intro. subst. apply lt_irrefl in H. auto. Qed.
Proof. intros. intro. subst. apply lt_irrefl in H. auto. Qed.
a < b -> ~ (b < a)
Lemma lt_not_lt : forall a b : tA, a < b -> ~ (b < a).
Proof. intros. intro. apply lt_gt_contr in H; auto. Qed.
End lt.
Proof. intros. intro. apply lt_gt_contr in H; auto. Qed.
End lt.
About "le"
Section le.
Context `{HOrder : Order}.
Notation "a >= b" := (Ale b a).
Notation "a > b" := (Alt b a).
Infix "<" := Alt.
Infix "<=" := Ale.
Lemma le_if_lt : forall a b : tA, a < b -> a <= b.
Proof. intros. apply lt_le_cong. auto. Qed.
Lemma lt_if_le_and_neq : forall a b : tA, a <= b -> a <> b -> a < b.
Proof. intros. apply lt_le_cong in H. destruct H; auto. easy. Qed.
Context `{HOrder : Order}.
Notation "a >= b" := (Ale b a).
Notation "a > b" := (Alt b a).
Infix "<" := Alt.
Infix "<=" := Ale.
Lemma le_if_lt : forall a b : tA, a < b -> a <= b.
Proof. intros. apply lt_le_cong. auto. Qed.
Lemma lt_if_le_and_neq : forall a b : tA, a <= b -> a <> b -> a < b.
Proof. intros. apply lt_le_cong in H. destruct H; auto. easy. Qed.
a <= a
a <= b -> b <= a -> a = b
Lemma le_antisym : forall a b : tA, a <= b -> b <= a -> a = b.
Proof.
intros. apply lt_le_cong in H, H0. destruct H, H0; subst; auto.
apply lt_antisym; auto.
Qed.
Proof.
intros. apply lt_le_cong in H, H0. destruct H, H0; subst; auto.
apply lt_antisym; auto.
Qed.
a <= b -> b <= c -> a <= c
Lemma le_trans : forall a b c : tA, a <= b -> b <= c -> a <= c.
Proof.
intros. apply lt_le_cong in H, H0. apply lt_le_cong.
destruct H, H0; subst; auto. left. apply lt_trans with b; auto.
Qed.
Proof.
intros. apply lt_le_cong in H, H0. apply lt_le_cong.
destruct H, H0; subst; auto. left. apply lt_trans with b; auto.
Qed.
a < b -> b <= c -> a < c
Lemma lt_trans_lt_le : forall a b c : tA, a < b -> b <= c -> a < c.
Proof.
intros. pose proof (le_if_lt H). pose proof (le_trans H1 H0).
apply lt_if_le_and_neq; auto.
intro. subst. pose proof (le_antisym H0 H1). subst. apply lt_irrefl in H; auto.
Qed.
Proof.
intros. pose proof (le_if_lt H). pose proof (le_trans H1 H0).
apply lt_if_le_and_neq; auto.
intro. subst. pose proof (le_antisym H0 H1). subst. apply lt_irrefl in H; auto.
Qed.
a <= b -> b < c -> a < c
Lemma lt_trans_le_lt : forall a b c : tA, a <= b -> b < c -> a < c.
Proof.
intros. pose proof (le_if_lt H0). pose proof (le_trans H H1).
apply lt_if_le_and_neq; auto.
intro. subst. pose proof (le_antisym H H1). subst. apply lt_irrefl in H0; auto.
Qed.
Proof.
intros. pose proof (le_if_lt H0). pose proof (le_trans H H1).
apply lt_if_le_and_neq; auto.
intro. subst. pose proof (le_antisym H H1). subst. apply lt_irrefl in H0; auto.
Qed.
a < b -> ~ (b <= a)
Lemma lt_not_le : forall a b : tA, a < b -> ~ (b <= a).
Proof.
intros. rewrite lt_le_cong. apply and_not_or. split.
apply lt_not_lt; auto. symmetry. apply lt_not_eq; auto.
Qed.
Proof.
intros. rewrite lt_le_cong. apply and_not_or. split.
apply lt_not_lt; auto. symmetry. apply lt_not_eq; auto.
Qed.
~ (a <= b) -> b < a
Lemma not_le_lt : forall a b : tA, ~ (a <= b) -> b < a.
Proof.
intros. rewrite lt_le_cong in H. apply not_or_and in H. destruct H.
destruct (lt_cases a b) as [[H1|H1]|H1]; try easy.
Qed.
Proof.
intros. rewrite lt_le_cong in H. apply not_or_and in H. destruct H.
destruct (lt_cases a b) as [[H1|H1]|H1]; try easy.
Qed.
a <= b -> ~ (b < a)
Lemma le_not_lt : forall a b : tA, a <= b -> ~ (b < a).
Proof.
intros. rewrite lt_le_cong in H. destruct H.
apply lt_not_lt; auto. subst. apply lt_irrefl.
Qed.
Proof.
intros. rewrite lt_le_cong in H. destruct H.
apply lt_not_lt; auto. subst. apply lt_irrefl.
Qed.
~ (a < b) -> b <= a
Lemma not_lt_le : forall a b : tA, ~ (a < b) -> b <= a.
Proof.
intros. apply lt_le_cong.
destruct (lt_cases a b) as [[H1|H1]|H1]; auto; try easy.
Qed.
Proof.
intros. apply lt_le_cong.
destruct (lt_cases a b) as [[H1|H1]|H1]; auto; try easy.
Qed.
a <= b -> b <= a -> a = b
Lemma le_ge_eq : forall a b : tA, a <= b -> b <= a -> a = b.
Proof.
intros. apply le_not_lt in H,H0.
destruct (lt_cases a b) as [[H1|H1]|H1]; easy.
Qed.
Proof.
intros. apply le_not_lt in H,H0.
destruct (lt_cases a b) as [[H1|H1]|H1]; easy.
Qed.
{a <= b} + {~(a <= b)}
Lemma Ale_dec : forall a b : tA, {a <= b} + {~(a <= b)}.
Proof.
intros. destruct (Alt_dec b a); auto.
- right. apply lt_not_le; auto.
- left. apply not_lt_le; auto.
Qed.
#[export] Instance Ale_Dec : Dec Ale.
Proof. constructor; intros. apply Ale_dec. Qed.
Proof.
intros. destruct (Alt_dec b a); auto.
- right. apply lt_not_le; auto.
- left. apply not_lt_le; auto.
Qed.
#[export] Instance Ale_Dec : Dec Ale.
Proof. constructor; intros. apply Ale_dec. Qed.
a <= b \/ b < a
Lemma le_connected : forall a b : tA, a <= b \/ b < a.
Proof. intros. destruct (Ale_dec a b); auto. apply not_le_lt in n. auto. Qed.
End le.
Proof. intros. destruct (Ale_dec a b); auto. apply not_le_lt in n. auto. Qed.
End le.
Simplify expressions on Order {=, <, <=}
Ltac order_core lt le :=
intros;
repeat
(try
match goal with
| H: Order lt le |- {?a = ?b} + {?a <> ?b} => apply Aeq_dec
| H: Order lt le |- {lt ?a ?b} + {~(lt ?a ?b)} => apply Alt_dec
| H: Order lt le |- {le ?a ?b} + {~(le ?a ?b)} => apply Ale_dec
| H: Order lt le |- le ?a ?a => apply le_refl
| H: Order lt le |- ~(lt ?a ?a) => apply lt_irrefl
| H: Order lt le, H1: ?a <> ?b, H2: le ?a ?b |- lt ?a ?b =>
apply lt_if_le_and_neq
end;
auto; try easy).
intros;
repeat
(try
match goal with
| H: Order lt le |- {?a = ?b} + {?a <> ?b} => apply Aeq_dec
| H: Order lt le |- {lt ?a ?b} + {~(lt ?a ?b)} => apply Alt_dec
| H: Order lt le |- {le ?a ?b} + {~(le ?a ?b)} => apply Ale_dec
| H: Order lt le |- le ?a ?a => apply le_refl
| H: Order lt le |- ~(lt ?a ?a) => apply lt_irrefl
| H: Order lt le, H1: ?a <> ?b, H2: le ?a ?b |- lt ?a ?b =>
apply lt_if_le_and_neq
end;
auto; try easy).
Simplify expressions on Order in current context
Section test.
Context `{Order}.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
Variable a b c d e f g : tA.
Goal {a = b} + {a <> b}. order. Qed.
Goal {a < b} + {~(a < b)}. order. Qed.
Goal {a <= b} + {~(a <= b)}. order. Qed.
Goal a <= a. order. Qed.
Goal ~(a < a). order. Qed.
Goal a <> b -> a <= b -> a < b. order. Qed.
End test.
Context `{Order}.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
Variable a b c d e f g : tA.
Goal {a = b} + {a <> b}. order. Qed.
Goal {a < b} + {~(a < b)}. order. Qed.
Goal {a <= b} + {~(a <= b)}. order. Qed.
Goal a <= a. order. Qed.
Goal ~(a < a). order. Qed.
Goal a <> b -> a <= b -> a < b. order. Qed.
End test.
Class OrderedARing {tA} Aadd Azero Aopp Amul Aone Alt Ale := {
or_Ring :: ARing Aadd Azero Aopp Amul Aone;
or_Order :: Order Alt Ale;
lt_add_compat_r : forall a b c : tA,
Alt a b -> Alt (Aadd a c) (Aadd b c);
lt_mul_compat_r : forall a b c : tA,
Alt a b -> Alt Azero c -> Alt (Amul a c) (Amul b c);
}.
Coercion or_Ring : OrderedARing >-> ARing.
Coercion or_Order : OrderedARing >-> Order.
Section theories.
Context `{HOrderedARing : OrderedARing}.
Add Ring ring_inst : (make_ring_theory HOrderedARing).
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + - b).
Infix "-" := Asub : A_scope.
Infix "*" := Amul : A_scope.
Notation "1" := Aone : A_scope.
Notation "2" := (1 + 1) : A_scope.
Notation "a ²" := (a * a) : A_scope.
Notation "a > b" := (Alt b a) : A_scope.
Notation "a >= b" := (Ale b a) : A_scope.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
Context `{HOrderedARing : OrderedARing}.
Add Ring ring_inst : (make_ring_theory HOrderedARing).
Infix "+" := Aadd : A_scope.
Notation "0" := Azero : A_scope.
Notation "- a" := (Aopp a) : A_scope.
Notation Asub a b := (a + - b).
Infix "-" := Asub : A_scope.
Infix "*" := Amul : A_scope.
Notation "1" := Aone : A_scope.
Notation "2" := (1 + 1) : A_scope.
Notation "a ²" := (a * a) : A_scope.
Notation "a > b" := (Alt b a) : A_scope.
Notation "a >= b" := (Ale b a) : A_scope.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
Lemma lt_add_compat_l : forall a b c : tA, a < b -> c + a < c + b.
Proof. intros. rewrite !(commutative c _). apply lt_add_compat_r; auto. Qed.
Lemma lt_add_compat : forall a b c d : tA, a < b -> c < d -> a + c < b + d.
Proof.
intros. apply lt_trans with (a + d).
apply lt_add_compat_l; auto. apply lt_add_compat_r; auto.
Qed.
Lemma lt_add_reg_l : forall a b c, c + a < c + b -> a < b.
Proof.
intros. apply lt_add_compat_l with (c:=-c) in H.
rewrite <- !associative in H. rewrite inverseLeft in H.
rewrite !identityLeft in H. auto.
Qed.
Lemma lt_add_reg_r : forall a b c, a + c < b + c -> a < b.
Proof.
intros.
apply lt_add_compat_r with (c:=-c) in H.
rewrite !associative in H. rewrite inverseRight in H.
rewrite !identityRight in H. auto.
Qed.
Lemma lt_opp_iff : forall a b : tA, a < b <-> - b < - a.
Proof.
intros. split; intros.
- apply lt_add_compat_l with (c:=-b) in H.
apply lt_add_compat_r with (c:=-a) in H.
replace (-b + a + -a) with (-b) in H; [|ring].
replace (-b + b + -a) with (-a) in H; [|ring]. auto.
- apply lt_add_reg_l with (c:=-b).
apply lt_add_reg_r with (c:=-a).
replace (-b + a + -a) with (-b); [|ring].
replace (-b + b + -a) with (-a); [|ring]. auto.
Qed.
Lemma lt_sub_iff : forall a b : tA, a < b <-> a - b < 0.
Proof.
intros. split; intros.
- apply lt_add_reg_r with (c := b). rewrite identityLeft at 1.
rewrite associative. rewrite inverseLeft. rewrite identityRight. auto.
- apply lt_add_compat_r with (c := b) in H.
rewrite identityLeft in H at 1.
rewrite associative in H. rewrite inverseLeft in H.
rewrite identityRight in H. auto.
Qed.
a < 0 <-> 0 < - a
Lemma lt0_iff_neg : forall a : tA, a < 0 <-> 0 < - a.
Proof. intros. rewrite lt_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
Proof. intros. rewrite lt_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
0 < a <-> - a < 0
Lemma gt0_iff_neg : forall a : tA, 0 < a <-> - a < 0.
Proof. intros. rewrite lt_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
Proof. intros. rewrite lt_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
0 < a -> 0 < b -> 0 < a + b
Lemma add_gt0_if_gt0_gt0 : forall a b : tA, 0 < a -> 0 < b -> 0 < a + b.
Proof.
intros. replace 0 with (0 + 0) by ring. apply lt_add_compat; auto.
Qed.
Proof.
intros. replace 0 with (0 + 0) by ring. apply lt_add_compat; auto.
Qed.
a < 0 -> b < 0 -> a + b < 0
Lemma add_lt0_if_lt0_lt0 : forall a b : tA, a < 0 -> b < 0 -> a + b < 0.
Proof.
intros. replace 0 with (0 + 0) by ring. apply lt_add_compat; auto.
Qed.
Proof.
intros. replace 0 with (0 + 0) by ring. apply lt_add_compat; auto.
Qed.
a < 0 -> 0 < b -> a < b
Lemma lt_if_lt0_gt0 : forall a b : tA, a < 0 -> 0 < b -> a < b.
Proof. intros. apply lt_trans with 0; auto. Qed.
Proof. intros. apply lt_trans with 0; auto. Qed.
a <= 0 -> 0 < b -> a < b
Lemma lt_if_le0_gt0 : forall a b : tA, a <= 0 -> 0 < b -> a < b.
Proof.
intros. destruct (Aeqdec a 0). subst; auto.
apply lt_trans with 0; auto. apply lt_le_cong in H. destruct H; easy.
Qed.
Proof.
intros. destruct (Aeqdec a 0). subst; auto.
apply lt_trans with 0; auto. apply lt_le_cong in H. destruct H; easy.
Qed.
a < 0 -> 0 <= b -> a < b
Lemma lt_if_lt0_ge0 : forall a b : tA, a < 0 -> 0 <= b -> a < b.
Proof.
intros. destruct (Aeqdec b 0). subst; auto.
apply lt_trans with 0; auto. apply lt_le_cong in H0. destruct H0; auto.
symmetry in H0. easy.
Qed.
Proof.
intros. destruct (Aeqdec b 0). subst; auto.
apply lt_trans with 0; auto. apply lt_le_cong in H0. destruct H0; auto.
symmetry in H0. easy.
Qed.
a < 0 -> a < - a
Lemma lt_neg_r_if_lt0 : forall a : tA, a < 0 -> a < - a.
Proof. intros. apply lt_if_lt0_gt0; auto. apply lt0_iff_neg; auto. Qed.
Proof. intros. apply lt_if_lt0_gt0; auto. apply lt0_iff_neg; auto. Qed.
0 < a -> - a < a
Lemma gt_neg_l_if_gt0 : forall a : tA, 0 < a -> - a < a.
Proof. intros. apply lt_if_lt0_gt0; auto. apply gt0_iff_neg; auto. Qed.
Proof. intros. apply lt_if_lt0_gt0; auto. apply gt0_iff_neg; auto. Qed.
Lemma lt_mul_compat_l : forall a b c : tA, a < b -> 0 < c -> c * a < c * b.
Proof. intros. rewrite !(commutative c _). apply lt_mul_compat_r; auto. Qed.
Lemma lt_mul_compat_l_neg : forall a b c : tA, a < b -> c < 0 -> c * b < c * a.
Proof.
intros. apply lt_opp_iff in H0. rewrite group_opp_0 in H0.
apply lt_opp_iff. ring_simplify. apply lt_mul_compat_l; auto.
Qed.
Lemma lt_mul_compat_r_neg : forall a b c : tA, a < b -> c < 0 -> b * c < a * c.
Proof. intros. rewrite !(commutative _ c). apply lt_mul_compat_l_neg; auto. Qed.
0 < a -> 0 < b -> 0 < a * b
Lemma mul_gt0_if_gt0_gt0 : forall a b : tA, 0 < a -> 0 < b -> 0 < a * b.
Proof.
intros. replace 0 with (a * 0). apply lt_mul_compat_l; auto. ring.
Qed.
Proof.
intros. replace 0 with (a * 0). apply lt_mul_compat_l; auto. ring.
Qed.
a < 0 -> b < 0 -> 0 < a * b
Lemma mul_gt0_if_lt0_lt0 : forall a b : tA, a < 0 -> b < 0 -> 0 < a * b.
Proof.
intros. replace (a * b) with ((- a) * (- b)) by ring. apply mul_gt0_if_gt0_gt0.
apply lt0_iff_neg; auto. apply lt0_iff_neg; auto.
Qed.
Proof.
intros. replace (a * b) with ((- a) * (- b)) by ring. apply mul_gt0_if_gt0_gt0.
apply lt0_iff_neg; auto. apply lt0_iff_neg; auto.
Qed.
0 < a -> b < 0 -> a * b < 0
Lemma mul_lt0_if_gt0_lt0 : forall a b : tA, 0 < a -> b < 0 -> a * b < 0.
Proof.
intros. replace 0 with (a * 0). apply lt_mul_compat_l; auto. ring.
Qed.
Proof.
intros. replace 0 with (a * 0). apply lt_mul_compat_l; auto. ring.
Qed.
a < 0 -> 0 < b -> a * b < 0
Lemma mul_lt0_if_lt0_gt0 : forall a b : tA, a < 0 -> 0 < b -> a * b < 0.
Proof.
intros. replace 0 with (0 * b). apply lt_mul_compat_r; auto. ring.
Qed.
Proof.
intros. replace 0 with (0 * b). apply lt_mul_compat_r; auto. ring.
Qed.
0 < a * b -> 0 < a -> 0 < b
Lemma gt0_mul_reg_gt0 : forall a b : tA, 0 < a * b -> 0 < a -> 0 < b.
Proof.
intros. destruct (lt_cases 0 b) as [[H1|H1]|H1]; auto.
- apply lt0_iff_neg in H1.
pose proof (mul_gt0_if_gt0_gt0 H0 H1).
replace (a * - b) with (- (a * b)) in H2 by ring.
apply lt0_iff_neg in H2. apply lt_gt_contr in H; easy.
- rewrite <- H1 in *. ring_simplify in H. apply lt_irrefl in H; easy.
Qed.
Proof.
intros. destruct (lt_cases 0 b) as [[H1|H1]|H1]; auto.
- apply lt0_iff_neg in H1.
pose proof (mul_gt0_if_gt0_gt0 H0 H1).
replace (a * - b) with (- (a * b)) in H2 by ring.
apply lt0_iff_neg in H2. apply lt_gt_contr in H; easy.
- rewrite <- H1 in *. ring_simplify in H. apply lt_irrefl in H; easy.
Qed.
0 < a * b -> a < 0 -> b < 0
Lemma gt0_mul_reg_lt0 : forall a b : tA, 0 < a * b -> a < 0 -> b < 0.
Proof.
intros. replace (a * b) with ((- a) * (- b)) in H by ring.
apply lt0_iff_neg. apply lt0_iff_neg in H0.
apply (gt0_mul_reg_gt0 H) in H0; auto.
Qed.
Proof.
intros. replace (a * b) with ((- a) * (- b)) in H by ring.
apply lt0_iff_neg. apply lt0_iff_neg in H0.
apply (gt0_mul_reg_gt0 H) in H0; auto.
Qed.
Lemma le_add_compat_r : forall a b c : tA, a <= b -> a + c <= b + c.
Proof.
intros. apply lt_le_cong. apply lt_le_cong in H. destruct H; subst; auto.
left. apply lt_add_compat_r; auto.
Qed.
Lemma le_add_compat_l : forall a b c : tA, a <= b -> c + a <= c + b.
Proof. intros. rewrite !(commutative c _). apply le_add_compat_r; auto. Qed.
Lemma le_add_compat : forall a b c d : tA, a <= b -> c <= d -> a + c <= b + d.
Proof.
intros. apply le_trans with (a + d).
apply le_add_compat_l; auto. apply le_add_compat_r; auto.
Qed.
Lemma le_add_reg_l : forall a b c, c + a <= c + b -> a <= b.
Proof.
intros. apply le_add_compat_l with (c:=-c) in H.
rewrite <- !associative in H. rewrite inverseLeft in H.
rewrite (identityLeft a) in H. rewrite (identityLeft b) in H. auto.
Qed.
Lemma le_add_reg_r : forall a b c, a + c <= b + c -> a <= b.
Proof.
intros. apply le_add_compat_r with (c:=-c) in H.
rewrite !associative in H. rewrite inverseRight in H.
rewrite (identityRight a) in H. rewrite (identityRight b) in H. auto.
Qed.
0 <= a -> 0 <= b -> 0 <= a + b
Lemma add_ge0_if_ge0_ge0 : forall a b : tA, 0 <= a -> 0 <= b -> 0 <= a + b.
Proof.
intros. replace 0 with (0 + 0) by ring. apply le_add_compat; auto.
Qed.
Proof.
intros. replace 0 with (0 + 0) by ring. apply le_add_compat; auto.
Qed.
a <= 0 -> b <= 0 -> a + b <= 0
Lemma add_le0_if_le0_le0 : forall a b : tA, a <= 0 -> b <= 0 -> a + b <= 0.
Proof.
intros. replace 0 with (0 + 0) by ring. apply le_add_compat; auto.
Qed.
Lemma le_opp_iff : forall a b : tA, a <= b <-> - b <= - a.
Proof.
intros. rewrite !lt_le_cong. split; intros; destruct H; subst; auto.
- apply lt_opp_iff in H; auto.
- apply lt_opp_iff in H; auto.
- apply group_opp_inj in H. auto.
Qed.
Proof.
intros. replace 0 with (0 + 0) by ring. apply le_add_compat; auto.
Qed.
Lemma le_opp_iff : forall a b : tA, a <= b <-> - b <= - a.
Proof.
intros. rewrite !lt_le_cong. split; intros; destruct H; subst; auto.
- apply lt_opp_iff in H; auto.
- apply lt_opp_iff in H; auto.
- apply group_opp_inj in H. auto.
Qed.
a <= 0 <-> 0 <= - a
Lemma le0_iff_neg : forall a : tA, a <= 0 <-> 0 <= - a.
Proof. intros. rewrite le_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
Proof. intros. rewrite le_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
0 <= a <-> - a <= 0
Lemma ge0_iff_neg : forall a : tA, 0 <= a <-> - a <= 0.
Proof. intros. rewrite le_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
Proof. intros. rewrite le_opp_iff. rewrite group_opp_0 at 1. easy. Qed.
a <= 0 -> 0 <= b -> a <= b
Lemma le_if_le0_ge0 : forall a b : tA, a <= 0 -> 0 <= b -> a <= b.
Proof. intros. apply le_trans with 0; auto. Qed.
Proof. intros. apply le_trans with 0; auto. Qed.
a < 0 -> 0 < b -> a <= b
Lemma le_if_lt0_gt0 : forall a b : tA, a < 0 -> 0 < b -> a <= b.
Proof. intros. apply le_if_lt. apply lt_if_lt0_gt0; auto. Qed.
Proof. intros. apply le_if_lt. apply lt_if_lt0_gt0; auto. Qed.
a <= 0 -> 0 < b -> a <= b
Lemma le_if_le0_gt0 : forall a b : tA, a <= 0 -> 0 < b -> a <= b.
Proof. intros. apply le_if_lt. apply lt_if_le0_gt0; auto. Qed.
Proof. intros. apply le_if_lt. apply lt_if_le0_gt0; auto. Qed.
a < 0 -> 0 <= b -> a <= b
Lemma le_if_lt0_ge0 : forall a b : tA, a < 0 -> 0 <= b -> a <= b.
Proof. intros. apply le_if_lt. apply lt_if_lt0_ge0; auto. Qed.
Proof. intros. apply le_if_lt. apply lt_if_lt0_ge0; auto. Qed.
a <= 0 -> a <= - a
Lemma le_neg_r_if_le0 : forall a : tA, a <= 0 -> a <= - a.
Proof. intros. apply le_if_le0_ge0; auto. apply le0_iff_neg; auto. Qed.
Proof. intros. apply le_if_le0_ge0; auto. apply le0_iff_neg; auto. Qed.
a < 0 -> a <= - a
Lemma le_neg_r_if_lt0 : forall a : tA, a < 0 -> a <= - a.
Proof. intros. apply le_neg_r_if_le0. apply le_if_lt; auto. Qed.
Proof. intros. apply le_neg_r_if_le0. apply le_if_lt; auto. Qed.
0 <= a -> - a <= a
Lemma ge_neg_l_if_ge0 : forall a : tA, 0 <= a -> - a <= a.
Proof. intros. apply le_if_le0_ge0; auto. apply ge0_iff_neg; auto. Qed.
Proof. intros. apply le_if_le0_ge0; auto. apply ge0_iff_neg; auto. Qed.
0 < a -> - a <= a
Lemma ge_neg_l_if_gt0 : forall a : tA, 0 < a -> - a <= a.
Proof. intros. apply ge_neg_l_if_ge0. apply le_if_lt; auto. Qed.
Proof. intros. apply ge_neg_l_if_ge0. apply le_if_lt; auto. Qed.
Lemma le_mul_compat_r : forall a b c : tA, a <= b -> 0 <= c -> a * c <= b * c.
Proof.
intros. apply lt_le_cong. apply lt_le_cong in H, H0. destruct H,H0; auto.
- left. apply lt_mul_compat_r; auto.
- rewrite <- H0. rewrite (ring_mul_0_r a). rewrite (ring_mul_0_r b). auto.
- subst; auto.
- subst; auto.
Qed.
Lemma le_mul_compat_l : forall a b c : tA, a <= b -> 0 <= c -> c * a <= c * b.
Proof. intros. rewrite !(commutative c _). apply le_mul_compat_r; auto. Qed.
Lemma le_mul_compat_l_neg : forall a b c : tA, a <= b -> c < 0 -> c * b <= c * a.
Proof.
intros. apply lt_le_cong in H. destruct H.
- apply lt_mul_compat_l_neg with (c:=c) in H; auto. apply le_if_lt; auto.
- subst. apply le_refl.
Qed.
Lemma le_mul_compat_r_neg : forall a b c : tA, a <= b -> c < 0 -> b * c <= a * c.
Proof. intros. rewrite !(commutative _ c). apply le_mul_compat_l_neg; auto. Qed.
0 <= a -> 0 <= b -> 0 <= a * b
Lemma mul_ge0_if_ge0_ge0 : forall a b : tA, 0 <= a -> 0 <= b -> 0 <= a * b.
Proof.
intros. replace 0 with (a * 0). apply le_mul_compat_l; auto. ring.
Qed.
Proof.
intros. replace 0 with (a * 0). apply le_mul_compat_l; auto. ring.
Qed.
a <= 0 -> b <= 0 -> 0 <= a * b
Lemma mul_ge0_if_le0_le0 : forall a b : tA, a <= 0 -> b <= 0 -> 0 <= a * b.
Proof.
intros.
intros. replace (a * b) with ((- a) * (- b)) by ring. apply mul_ge0_if_ge0_ge0.
apply le0_iff_neg; auto. apply le0_iff_neg; auto.
Qed.
Proof.
intros.
intros. replace (a * b) with ((- a) * (- b)) by ring. apply mul_ge0_if_ge0_ge0.
apply le0_iff_neg; auto. apply le0_iff_neg; auto.
Qed.
0 <= a -> b <= 0 -> a * b <= 0
Lemma mul_le0_if_ge0_le0 : forall a b : tA, 0 <= a -> b <= 0 -> a * b <= 0.
Proof.
intros. replace 0 with (a * 0). apply le_mul_compat_l; auto. ring.
Qed.
Proof.
intros. replace 0 with (a * 0). apply le_mul_compat_l; auto. ring.
Qed.
a <= 0 -> 0 <= b -> a * b <= 0
Lemma mul_le0_if_le0_ge0 : forall a b : tA, a <= 0 -> 0 <= b -> a * b <= 0.
Proof.
intros. replace 0 with (0 * b). apply le_mul_compat_r; auto. ring.
Qed.
Proof.
intros. replace 0 with (0 * b). apply le_mul_compat_r; auto. ring.
Qed.
0 <= a * b -> 0 < a -> 0 <= b
Lemma ge0_mul_reg_ge0 : forall a b : tA, 0 <= a * b -> 0 < a -> 0 <= b.
Proof.
intros. destruct (Ale_dec 0 b) as [H1|H1]; auto.
destruct (Aeqdec b 0) as [H2|H2].
- rewrite H2 in *. apply le_refl.
- exfalso. apply not_le_lt in H1.
pose proof (mul_lt0_if_gt0_lt0 H0 H1).
apply lt_not_le in H3. easy.
Qed.
Proof.
intros. destruct (Ale_dec 0 b) as [H1|H1]; auto.
destruct (Aeqdec b 0) as [H2|H2].
- rewrite H2 in *. apply le_refl.
- exfalso. apply not_le_lt in H1.
pose proof (mul_lt0_if_gt0_lt0 H0 H1).
apply lt_not_le in H3. easy.
Qed.
0 <= a * b -> a < 0 -> b <= 0
Lemma ge0_mul_reg_le0 : forall a b : tA, 0 <= a * b -> a < 0 -> b <= 0.
Proof.
intros. replace (a * b) with ((- a) * (- b)) in H by ring.
apply lt0_iff_neg in H0. apply ge0_mul_reg_ge0 in H; auto.
apply le0_iff_neg; auto.
Qed.
Proof.
intros. replace (a * b) with ((- a) * (- b)) in H by ring.
apply lt0_iff_neg in H0. apply ge0_mul_reg_ge0 in H; auto.
apply le0_iff_neg; auto.
Qed.
Lemma sqr_ge0 : forall a : tA, a * a >= 0.
Proof.
intros.
pose proof (lt_connected 0 a). destruct H as [H|[H|H]].
- apply le_if_lt.
pose proof (lt_mul_compat_r H H). rewrite ring_mul_0_l in H0; auto.
- apply le_if_lt. rewrite lt_opp_iff in H. rewrite group_opp_0 in H.
rewrite <- (group_opp_opp a). rewrite ring_mul_opp_opp.
pose proof (lt_mul_compat_r H H). rewrite ring_mul_0_l in H0. auto.
- rewrite <- H. rewrite (ring_mul_0_l 0). apply le_refl.
Qed.
Lemma add_eq0_imply_0_l : forall a b : tA, 0 <= a -> 0 <= b -> a + b = 0 -> a = 0.
Proof.
intros. apply lt_le_cong in H, H0.
destruct H as [H|H], H0 as [H0|H0]; auto.
- assert (a + b > 0 + 0).
+ apply lt_add_compat; auto.
+ rewrite H1 in H2. rewrite identityLeft in H2. apply lt_irrefl in H2. easy.
- rewrite <- H0 in *. rewrite identityRight in H1. auto.
Qed.
Lemma add_eq0_imply_0_r : forall a b : tA, 0 <= a -> 0 <= b -> a + b = 0 -> b = 0.
Proof.
intros. rewrite commutative in H1. apply add_eq0_imply_0_l in H1; auto.
Qed.
0 <= a + b -> a >= - b
Lemma sub_ge0_imply_ge : forall a b : tA, 0 <= a - b -> a >= b.
Proof.
intros. apply le_add_compat_r with (c:= b) in H. ring_simplify in H. auto.
Qed.
Proof.
intros. apply le_add_compat_r with (c:= b) in H. ring_simplify in H. auto.
Qed.
2 * a * b <= a² + b²
Lemma mul_le_add_sqr : forall a b : tA, 2 * a * b <= a² + b².
Proof.
intros.
assert (0 <= (a - b)²). apply sqr_ge0.
ring_simplify in H.
replace (a ² + - (2 * a * b) + b ²)
with (a ² + b ² - (2 * a * b)) in H by ring.
apply sub_ge0_imply_ge in H. auto.
Qed.
Proof.
intros.
assert (0 <= (a - b)²). apply sqr_ge0.
ring_simplify in H.
replace (a ² + - (2 * a * b) + b ²)
with (a ² + b ² - (2 * a * b)) in H by ring.
apply sub_ge0_imply_ge in H. auto.
Qed.
Definition Aabs (a : tA) : tA := if Ale_dec 0 a then a else - a.
Notation "| a |" := (Aabs a) : A_scope.
0 <= a -> | a | = a
Lemma Aabs_right : forall a : tA, 0 <= a -> | a | = a.
Proof. intros. unfold Aabs. destruct (Ale_dec 0 a); auto. easy. Qed.
Proof. intros. unfold Aabs. destruct (Ale_dec 0 a); auto. easy. Qed.
a < 0 -> | a | = - a
Lemma Aabs_left : forall a : tA, a < 0 -> | a | = - a.
Proof.
intros. unfold Aabs. destruct (Ale_dec 0 a); auto.
apply lt_not_le in H. easy.
Qed.
Proof.
intros. unfold Aabs. destruct (Ale_dec 0 a); auto.
apply lt_not_le in H. easy.
Qed.
| a * b | = | a | * | b |
Lemma Aabs_mul : forall a b : tA, | a * b | = | a | * | b |.
Proof.
intros. unfold Aabs.
destruct (Ale_dec 0 (a*b)), (Ale_dec 0 a), (Ale_dec 0 b); try ring.
- apply not_le_lt in n.
rewrite commutative in a0. apply ge0_mul_reg_le0 in a0; auto.
apply le_ge_eq in a1; auto. subst. ring.
- apply not_le_lt in n. apply ge0_mul_reg_le0 in a0; auto.
apply le_ge_eq in a1; auto. subst. ring.
- pose proof (mul_ge0_if_ge0_ge0 a0 a1). easy.
- apply not_le_lt in n,n0,n1. pose proof (mul_gt0_if_lt0_lt0 n0 n1).
exfalso. apply (lt_gt_contr n H).
Qed.
Proof.
intros. unfold Aabs.
destruct (Ale_dec 0 (a*b)), (Ale_dec 0 a), (Ale_dec 0 b); try ring.
- apply not_le_lt in n.
rewrite commutative in a0. apply ge0_mul_reg_le0 in a0; auto.
apply le_ge_eq in a1; auto. subst. ring.
- apply not_le_lt in n. apply ge0_mul_reg_le0 in a0; auto.
apply le_ge_eq in a1; auto. subst. ring.
- pose proof (mul_ge0_if_ge0_ge0 a0 a1). easy.
- apply not_le_lt in n,n0,n1. pose proof (mul_gt0_if_lt0_lt0 n0 n1).
exfalso. apply (lt_gt_contr n H).
Qed.
| a + b | <= | a | + | b |
Lemma Aabs_add_le : forall a b : tA, | a + b | <= | a | + | b |.
Proof.
intros. unfold Aabs.
destruct (Ale_dec 0 a), (Ale_dec 0 b), (Ale_dec 0 (a + b));
ring_simplify.
- apply le_refl.
- apply le_add_compat; apply ge_neg_l_if_ge0; auto.
- apply le_add_compat_l. apply le_neg_r_if_lt0; auto. apply not_le_lt; auto.
- apply le_add_compat_r. apply ge_neg_l_if_ge0; auto.
- apply le_add_compat_r. apply le_neg_r_if_lt0. apply not_le_lt; auto.
- apply le_add_compat_l. apply ge_neg_l_if_ge0; auto.
- apply not_le_lt in n,n0. pose proof (add_lt0_if_lt0_lt0 n n0).
apply le_not_lt in a0. easy.
- apply le_refl.
Qed.
End theories.
Proof.
intros. unfold Aabs.
destruct (Ale_dec 0 a), (Ale_dec 0 b), (Ale_dec 0 (a + b));
ring_simplify.
- apply le_refl.
- apply le_add_compat; apply ge_neg_l_if_ge0; auto.
- apply le_add_compat_l. apply le_neg_r_if_lt0; auto. apply not_le_lt; auto.
- apply le_add_compat_r. apply ge_neg_l_if_ge0; auto.
- apply le_add_compat_r. apply le_neg_r_if_lt0. apply not_le_lt; auto.
- apply le_add_compat_l. apply ge_neg_l_if_ge0; auto.
- apply not_le_lt in n,n0. pose proof (add_lt0_if_lt0_lt0 n n0).
apply le_not_lt in a0. easy.
- apply le_refl.
Qed.
End theories.
Class OrderedField {tA} Aadd Azero Aopp Amul Aone Ainv Alt Ale := {
of_Field :: @Field tA Aadd Azero Aopp Amul Aone Ainv;
of_OrderedRing :: @OrderedARing _ Aadd Azero Aopp Amul Aone Alt Ale;
}.
Coercion of_Field : OrderedField >-> Field.
Section Theory.
Context `{HOrderedField : OrderedField}.
Add Field field_inst : (make_field_theory HOrderedField).
Infix "+" := Aadd.
Notation "- a" := (Aopp a).
Notation "0" := Azero.
Infix "*" := Amul.
Notation "1" := Aone.
Notation "/ a" := (Ainv a).
Notation Adiv a b := (a * / b).
Infix "/" := Adiv.
Notation "a > b" := (Alt b a).
Notation "a >= b" := (Ale b a).
Infix "<" := Alt.
Infix "<=" := Ale.
Context `{HOrderedField : OrderedField}.
Add Field field_inst : (make_field_theory HOrderedField).
Infix "+" := Aadd.
Notation "- a" := (Aopp a).
Notation "0" := Azero.
Infix "*" := Amul.
Notation "1" := Aone.
Notation "/ a" := (Ainv a).
Notation Adiv a b := (a * / b).
Infix "/" := Adiv.
Notation "a > b" := (Alt b a).
Notation "a >= b" := (Ale b a).
Infix "<" := Alt.
Infix "<=" := Ale.
a <> 0 -> 0 < a * a
Lemma sqr_gt0 : forall a : tA, a <> 0 -> 0 < a * a.
Proof.
intros. apply lt_if_le_and_neq.
- apply sqr_ge0.
- intro. symmetry in H0. apply field_sqr_eq0_reg in H0. easy.
Qed.
Proof.
intros. apply lt_if_le_and_neq.
- apply sqr_ge0.
- intro. symmetry in H0. apply field_sqr_eq0_reg in H0. easy.
Qed.
0 < 1
Lemma lt_0_1 : 0 < 1.
Proof.
assert (1 <> 0). apply field_1_neq_0.
pose proof (sqr_gt0 H). rewrite identityLeft in H0. auto.
Qed.
Proof.
assert (1 <> 0). apply field_1_neq_0.
pose proof (sqr_gt0 H). rewrite identityLeft in H0. auto.
Qed.
0 <= 1
0 < a -> 0 < / a
Lemma inv_gt0 : forall a : tA, 0 < a -> 0 < / a.
Proof.
intros.
assert (0 < a * / a).
{ field_simplify. apply lt_0_1. symmetry. apply lt_not_eq; auto. }
apply (gt0_mul_reg_gt0 H0); auto.
Qed.
Proof.
intros.
assert (0 < a * / a).
{ field_simplify. apply lt_0_1. symmetry. apply lt_not_eq; auto. }
apply (gt0_mul_reg_gt0 H0); auto.
Qed.
a < 0 -> / a < 0
Lemma inv_lt0 : forall a : tA, a < 0 -> / a < 0.
Proof.
intros. pose proof (lt_imply_neq H).
rewrite lt_opp_iff in H. rewrite group_opp_0 in H.
apply inv_gt0 in H. rewrite field_inv_opp in H; auto. apply lt0_iff_neg; auto.
Qed.
Proof.
intros. pose proof (lt_imply_neq H).
rewrite lt_opp_iff in H. rewrite group_opp_0 in H.
apply inv_gt0 in H. rewrite field_inv_opp in H; auto. apply lt0_iff_neg; auto.
Qed.
0 < a -> 0 < b -> a < b -> / b < / a
Lemma lt_inv : forall a b : tA, 0 < a -> 0 < b -> a < b -> / b < / a.
Proof.
intros.
apply lt_mul_compat_l with (c:=/b) in H1.
apply lt_mul_compat_r with (c:=/a) in H1.
rewrite field_mulInvL in H1. rewrite identityLeft in H1 at 1.
rewrite associative in H1. rewrite field_mulInvR, identityRight in H1. auto.
symmetry; apply lt_imply_neq; auto.
symmetry; apply lt_imply_neq; auto.
apply inv_gt0; auto.
apply inv_gt0; auto.
Qed.
Proof.
intros.
apply lt_mul_compat_l with (c:=/b) in H1.
apply lt_mul_compat_r with (c:=/a) in H1.
rewrite field_mulInvL in H1. rewrite identityLeft in H1 at 1.
rewrite associative in H1. rewrite field_mulInvR, identityRight in H1. auto.
symmetry; apply lt_imply_neq; auto.
symmetry; apply lt_imply_neq; auto.
apply inv_gt0; auto.
apply inv_gt0; auto.
Qed.
0 < c -> c * a < c * b -> a < b
Lemma lt_mul_reg_l : forall a b c, 0 < c -> c * a < c * b -> a < b.
Proof.
intros. apply lt_mul_compat_l with (c:=/c) in H0.
- rewrite <- !associative in H0. rewrite field_mulInvL in H0.
+ rewrite !identityLeft in H0; auto.
+ symmetry. apply lt_imply_neq; auto.
- apply inv_gt0; auto.
Qed.
Proof.
intros. apply lt_mul_compat_l with (c:=/c) in H0.
- rewrite <- !associative in H0. rewrite field_mulInvL in H0.
+ rewrite !identityLeft in H0; auto.
+ symmetry. apply lt_imply_neq; auto.
- apply inv_gt0; auto.
Qed.
0 < c -> a * c < b * c -> a < b
Lemma lt_mul_reg_r : forall a b c, 0 < c -> a * c < b * c -> a < b.
Proof.
intros. rewrite !(commutative _ c) in H0. apply lt_mul_reg_l in H0; auto.
Qed.
Proof.
intros. rewrite !(commutative _ c) in H0. apply lt_mul_reg_l in H0; auto.
Qed.
0 < a -> a < b -> a / b < 1
Lemma lt_imply_div_lt_1 : forall a b : tA, 0 < a -> a < b -> a / b < 1.
Proof.
intros. replace 1 with (b / b).
apply lt_mul_compat_r; auto. apply inv_gt0. apply lt_trans with a; auto.
apply field_mulInvR. symmetry. apply lt_imply_neq. apply lt_trans with a; auto.
Qed.
Proof.
intros. replace 1 with (b / b).
apply lt_mul_compat_r; auto. apply inv_gt0. apply lt_trans with a; auto.
apply field_mulInvR. symmetry. apply lt_imply_neq. apply lt_trans with a; auto.
Qed.
0 < a -> a <= b -> a / b <= 1
Lemma le_imply_div_le_1 : forall a b : tA, 0 < a -> a <= b -> a / b <= 1.
Proof.
intros. apply lt_le_cong in H0. destruct H0.
- apply le_if_lt. apply lt_imply_div_lt_1; auto.
- subst. rewrite field_mulInvR. apply le_refl.
symmetry. apply lt_imply_neq; auto.
Qed.
End Theory.
Proof.
intros. apply lt_le_cong in H0. destruct H0.
- apply le_if_lt. apply lt_imply_div_lt_1; auto.
- subst. rewrite field_mulInvR. apply le_refl.
symmetry. apply lt_imply_neq; auto.
Qed.
End Theory.
Class A2R {tA} Aadd Azero Aopp Amul Aone Ainv Alt Ale
(a2r : tA -> R) := {
a2r_add : forall a b : tA, a2r (Aadd a b) = (a2r a + a2r b)%R;
a2r_0 : a2r Azero = 0%R;
a2r_opp : forall a : tA, a2r (Aopp a) = (- (a2r a))%R;
a2r_mul : forall a b : tA, a2r (Amul a b) = (a2r a * a2r b)%R;
a2r_1 : a2r Aone = 1%R;
a2r_inv : forall a : tA, a <> Azero -> a2r (Ainv a) = (/ (a2r a))%R;
a2r_Order :: Order Alt Ale;
a2r_eq_iff : forall a b : tA, a2r a = a2r b <-> a = b;
a2r_lt_iff : forall a b : tA, (a2r a < a2r b)%R <-> Alt a b;
a2r_le_iff : forall a b : tA, (a2r a <= a2r b)%R <-> Ale a b
}.
(a2r : tA -> R) := {
a2r_add : forall a b : tA, a2r (Aadd a b) = (a2r a + a2r b)%R;
a2r_0 : a2r Azero = 0%R;
a2r_opp : forall a : tA, a2r (Aopp a) = (- (a2r a))%R;
a2r_mul : forall a b : tA, a2r (Amul a b) = (a2r a * a2r b)%R;
a2r_1 : a2r Aone = 1%R;
a2r_inv : forall a : tA, a <> Azero -> a2r (Ainv a) = (/ (a2r a))%R;
a2r_Order :: Order Alt Ale;
a2r_eq_iff : forall a b : tA, a2r a = a2r b <-> a = b;
a2r_lt_iff : forall a b : tA, (a2r a < a2r b)%R <-> Alt a b;
a2r_le_iff : forall a b : tA, (a2r a <= a2r b)%R <-> Ale a b
}.
Section Theory.
Context `{HA2R : A2R}.
Context {AeqDec : Dec (@eq tA)}.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
Context `{HA2R : A2R}.
Context {AeqDec : Dec (@eq tA)}.
Infix "<" := Alt : A_scope.
Infix "<=" := Ale : A_scope.
a2r a = 0 <-> a = 0
Lemma a2r_eq0_iff : forall a : tA, (a2r a = 0)%R <-> (a = Azero)%A.
Proof. intros. rewrite <- a2r_0. apply a2r_eq_iff. Qed.
Proof. intros. rewrite <- a2r_0. apply a2r_eq_iff. Qed.
a2r a = 1 <-> a = 1
Lemma a2r_eq1_iff : forall a : tA, (a2r a = 1)%R <-> (a = Aone)%A.
Proof. intros. rewrite <- a2r_1. apply a2r_eq_iff. Qed.
Proof. intros. rewrite <- a2r_1. apply a2r_eq_iff. Qed.
0 <= a2r a <-> 0 <= a
Lemma a2r_ge0_iff : forall a : tA, (0 <= a2r a)%R <-> (Azero <= a)%A.
Proof. intros. rewrite <- a2r_0. apply a2r_le_iff. Qed.
Proof. intros. rewrite <- a2r_0. apply a2r_le_iff. Qed.
0 < a2r a <-> 0 < a
Lemma a2r_gt0_iff : forall a : tA, (0 < a2r a)%R <-> (Azero < a)%A.
Proof. intros. rewrite <- a2r_0. apply a2r_lt_iff. Qed.
Section OrderedARing.
Context `{HOrderedARing :
OrderedARing tA Aadd Azero Aopp Amul Aone Alt Ale}.
Notation "| a |" := (Rabs a) : R_scope.
Notation "| a |" := (Aabs a) : A_scope.
Proof. intros. rewrite <- a2r_0. apply a2r_lt_iff. Qed.
Section OrderedARing.
Context `{HOrderedARing :
OrderedARing tA Aadd Azero Aopp Amul Aone Alt Ale}.
Notation "| a |" := (Rabs a) : R_scope.
Notation "| a |" := (Aabs a) : A_scope.
a2r | a | = | a2r a |
Lemma a2r_Aabs : forall a : tA, a2r (| a |) = | a2r a|%R.
Proof.
intros. unfold Aabs. destruct (Ale_dec Azero a).
- rewrite Rabs_right; auto. rewrite <- a2r_0.
apply Rle_ge. apply a2r_le_iff; auto.
- rewrite Rabs_left. rewrite a2r_opp. auto.
rewrite <- a2r_0. apply a2r_lt_iff. apply not_le_lt; auto.
Qed.
End OrderedARing.
End Theory.
Proof.
intros. unfold Aabs. destruct (Ale_dec Azero a).
- rewrite Rabs_right; auto. rewrite <- a2r_0.
apply Rle_ge. apply a2r_le_iff; auto.
- rewrite Rabs_left. rewrite a2r_opp. auto.
rewrite <- a2r_0. apply a2r_lt_iff. apply not_le_lt; auto.
Qed.
End OrderedARing.
End Theory.