module Algebra.Monoid where
Monoids🔗
A monoid is a semigroup equipped with a two-sided identity element: An element such that . For any particular choice of binary operator , if a two-sided identity exists, then it is unique; In this sense, “being a monoid” could be considered an “axiom” that semigroups may satisfy.
However, since semigroup homomorphisms do not automatically preserve
the identity element1, it is part of the type signature
for is-monoid
, being considered
structure that a semigroup may be equipped with.
record is-monoid (id : A) (_⋆_ : A → A → A) : Type (level-of A) where
field
: is-semigroup _⋆_
has-is-semigroup
open is-semigroup has-is-semigroup public
field
: {x : A} → id ⋆ x ≡ x
idl : {x : A} → x ⋆ id ≡ x
idr
open is-monoid public
The condition of defining a monoid is a proposition, so that we may safely decompose monoids as the structure , which has to satisfy the property of being a monoid.
private unquoteDecl eqv = declare-record-iso eqv (quote is-monoid)
instance
: ∀ {id : A} {_⋆_ : A → A → A} {n}
H-Level-is-monoid → H-Level (is-monoid id _⋆_) (suc n)
= prop-instance λ x →
H-Level-is-monoid let open is-monoid x in Iso→is-hlevel 1 eqv (hlevel 1) x
A monoid structure on
a type is given by the
choice of identity element, the choice of binary operation, and the
witness that these choices form a monoid. A Monoid
, then, is a type with
a monoid
structure.
record Monoid-on (A : Type ℓ) : Type ℓ where
field
: A
identity _⋆_ : A → A → A
: is-monoid identity _⋆_
has-is-monoid
open is-monoid has-is-monoid public
: (ℓ : Level) → Type (lsuc ℓ)
Monoid = Σ (Type ℓ) Monoid-on
Monoid ℓ
open Monoid-on
There is also a predicate which witnesses when an equivalence between monoids is a monoid homomorphism. It has to preserve the identity, and commute with the multiplication:
record
{ℓ ℓ'} {A : Type ℓ} {B : Type ℓ'}
Monoid-hom (s : Monoid-on A) (t : Monoid-on B)
(e : A → B)
: Type (ℓ ⊔ ℓ') where
private
module A = Monoid-on s
module B = Monoid-on t
field
: e A.identity ≡ B.identity
pres-id : (x y : A) → e (x A.⋆ y) ≡ e x B.⋆ e y
pres-⋆
open Monoid-hom
: (A B : Monoid ℓ) (e : A .fst ≃ B .fst) → Type _
Monoid≃ (e , _) = Monoid-hom (A .snd) (B .snd) e Monoid≃ A B
Relationships to unital magmas🔗
open import Algebra.Magma.Unital
By definition, every monoid is exactly a unital magma
that is also a semigroup
. However, adopting this as a
definition yields several issues especially when it comes to
metaprogramming, which is why this is instead expressed by explicitly
proving the implications between the properties.
First, we show that every monoid is a unital magma:
module _ {id : A} {_⋆_ : A → A → A} where
: is-monoid id _⋆_ → is-unital-magma id _⋆_
is-monoid→is-unital-magma .has-is-magma = mon .has-is-semigroup .has-is-magma
is-monoid→is-unital-magma mon .idl = mon .idl
is-monoid→is-unital-magma mon .idr = mon .idr is-monoid→is-unital-magma mon
“Reshuffling” the record fields also allows us to show the reverse direction, namely, that every unital semigroup is a monoid.
is-unital-magma→is-semigroup→is-monoid: is-unital-magma id _⋆_ → is-semigroup _⋆_ → is-monoid id _⋆_
.has-is-semigroup = sem
is-unital-magma→is-semigroup→is-monoid uni sem .idl = uni .idl
is-unital-magma→is-semigroup→is-monoid uni sem .idr = uni .idr is-unital-magma→is-semigroup→is-monoid uni sem
Inverses🔗
A useful application of the monoid laws is in showing that having an inverse is a property of a specific element, not structure on that element. To make this precise: Let be an element of a monoid, say ; If there are and such that and , then .
monoid-inverse-unique: ∀ {1M : A} {_⋆_ : A → A → A}
→ (m : is-monoid 1M _⋆_)
→ (e x y : A)
→ (x ⋆ e ≡ 1M) → (e ⋆ y ≡ 1M)
→ x ≡ y
This is a happy theorem where stating the assumptions takes as many lines as the proof, which is a rather boring calculation. Since is an identity for , we can freely multiply by to “sneak in” a :
{1M = 1M} {_⋆_} m e x y li1 ri2 =
monoid-inverse-unique (m .idr) ⟩
x ≡⟨ sym
x ⋆ ⌜ 1M ⌝ ≡˘⟨ ap¡ ri2 ⟩(e ⋆ y) ≡⟨ m .associative ⟩
x ⋆
⌜ x ⋆ e ⌝ ⋆ y ≡⟨ ap! li1 ⟩.idl ⟩
1M ⋆ y ≡⟨ m y ∎
Counterexample: The map which sends everything to zero is a semigroup homomorphism, but does not preserve the unit of .↩︎