module Order.Total where
open is-join
open is-meet
Total orders🔗
A total order is a partial order where every pair of elements can be compared.
record is-total-order {o ℓ} (P : Poset o ℓ) : Type (o ⊔ ℓ) where
open Poset P public
field
: ∀ x y → (x ≤ y) ⊎ (y ≤ x) compare
The ordering procedure, compare
, is orthogonal from having a decision procedure for the
order: if comparing
results in
we can not, in general, turn this into a proof of
Having a decision procedure for the relative order of two elements allows us to compute meets and joins in a very straightforward way: We test whether and, if that’s the case, then otherwise, The definition of is exactly dual.
module minmax {o ℓ} {P : Poset o ℓ} (to : is-total-order P) where
open is-total-order to
: (x y : ⌞ P ⌟) → ⌞ P ⌟
min with compare x y
min x y ... | inl p = x
... | inr q = y
The proofs that this is actually the meet of and proceed by doing the comparison again: this “unblocks” the computation of
abstract
: ∀ x y → min x y ≤ x
min-≤l with compare x y
min-≤l x y ... | inl p = ≤-refl
... | inr q = q
: ∀ x y → min x y ≤ y
min-≤r with compare x y
min-≤r x y ... | inl p = p
... | inr q = ≤-refl
: ∀ x y z → z ≤ x → z ≤ y → z ≤ min x y
min-univ with compare x y
min-univ x y z p q ... | inl _ = p
... | inr _ = q
This is everything we need to prove that we have our hands on a meet.
: ∀ x y → is-meet P x y (min x y)
min-is-meet .meet≤l = min-≤l x y
min-is-meet x y .meet≤r = min-≤r x y
min-is-meet x y .greatest = min-univ x y min-is-meet x y
Since the case of maxima is precisely dual, we present it with no further commentary.
: (x y : ⌞ P ⌟) → ⌞ P ⌟
max with compare x y
max x y ... | inl p = y
... | inr q = x
: ∀ x y → x ≤ max x y
max-≤l with compare x y
max-≤l x y ... | inl p = p
... | inr q = ≤-refl
: ∀ x y → y ≤ max x y
max-≤r with compare x y
max-≤r x y ... | inl p = ≤-refl
... | inr q = q
: ∀ x y z → x ≤ z → y ≤ z → max x y ≤ z
max-univ with compare x y
max-univ x y z p q ... | inl _ = q
... | inr _ = p
: ∀ x y → is-join P x y (max x y)
max-is-join .l≤join = max-≤l x y
max-is-join x y .r≤join = max-≤r x y
max-is-join x y .least = max-univ x y max-is-join x y
Decidable total orders🔗
In particularly nice cases, we can not only decide the relative position of a pair of elements, but whether a pair of elements is related at all: this is a decidable total order.
: ∀ {o ℓ} (P : Poset o ℓ) → Type _
is-decidable-poset = ∀ {x y} → Dec (x ≤ y)
is-decidable-poset P where open Poset P
record is-decidable-total-order {o ℓ} (P : Poset o ℓ) : Type (o ⊔ ℓ) where
field
: is-total-order P
has-is-total
open is-total-order has-is-total public
As the name implies, a decidable total order must be a total order; and it must also be decidable:
field
: is-decidable-poset P
⦃ dec-≤ ⦄ : Discrete ⌞ P ⌟ ⦃ discrete ⦄
Note that we have included a requirement that a decidable total order be discrete, i.e., that its equality is also decidable. This is purely for formalisation reasons — allowing the user to specify a more efficient decision procedure for equality —, since we can use the decidable ordering to decide equality.
private
: Discrete ⌞ P ⌟
was-discrete-anyways {x} {y} with holds? (x ≤ y) | holds? (y ≤ x)
was-discrete-anyways ... | yes x≤y | yes y≤x = yes (≤-antisym x≤y y≤x)
... | yes x≤y | no ¬y≤x = no λ x=y → ¬y≤x (≤-refl' (sym x=y))
... | no ¬x≤y | _ = no λ x=y → ¬x≤y (≤-refl' x=y)
Note that, if we have a decidable order, then the requirement of totality can be weakened to the property that, for any we have
which we refer to as weak totality.
: ∀ {x y} → ¬ (x ≤ y) → y ≤ x
from-not-≤ {x} {y} ¬x≤y with compare x y
from-not-≤ ... | inl x≤y = absurd (¬x≤y x≤y)
... | inr y≤x = y≤x
module _ {o ℓ} {P : Poset o ℓ} ⦃ _ : Discrete ⌞ P ⌟ ⦄ ⦃ _ : is-decidable-poset P ⦄ where
open Poset P
from-weakly-total: (wk : ∀ {x y} → ¬ (x ≤ y) → y ≤ x)
→ is-decidable-total-order P
= record { has-is-total = tot } where from-weakly-total wk
The construction is straightforward: we can test whether and if it holds, we’re done; Otherwise, weak totality lets us conclude that from the computed witness of
: ∀ x y → (x ≤ y) ⊎ (y ≤ x)
compare with holds? (x ≤ y)
compare x y ... | yes x≤y = inl x≤y
... | no ¬x≤y = inr (wk ¬x≤y)
: is-total-order P
tot = record { compare = compare } tot