open import 1Lab.Prelude

open import Algebra.Semigroup
module Algebra.Monoid where
private variable
  ℓ ℓ₁ : Level
  A : Type ℓ

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
    has-is-semigroup : is-semigroup __

  open is-semigroup has-is-semigroup public

  field
    idl : {x : A}  id ⋆ x ≡ x
    idr : {x : A}  x ⋆ id ≡ x

open is-monoid

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
  H-Level-is-monoid :  {id : A} {__ : A  A  A} {n}
                     H-Level (is-monoid id __) (suc n)
  H-Level-is-monoid = prop-instance λ x 
    let open is-monoid x in Iso→is-hlevel! 1 eqv 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
    identity : A
    __ : A  A  A

    has-is-monoid : is-monoid identity __

  open is-monoid has-is-monoid public

Monoid : (: Level)  Type (lsuc ℓ)
Monoid ℓ = Σ (Type ℓ) 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
  Monoid-hom {ℓ ℓ'} {A : Type ℓ} {B : Type ℓ'}
             (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
    pres-id : e A.identity ≡ B.identity
    pres-⋆ : (x y : A)  e (x A.⋆ y) ≡ e x B.⋆ e y

open Monoid-hom

Monoid≃ : (A B : Monoid ℓ) (e : A .fst ≃ B .fst)  Type _
Monoid≃ A B (e , _) = Monoid-hom (A .snd) (B .snd) e

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→is-unital-magma : is-monoid id __  is-unital-magma id __
  is-monoid→is-unital-magma mon .has-is-magma = mon .has-is-magma
  is-monoid→is-unital-magma mon .idl = mon .idl
  is-monoid→is-unital-magma mon .idr = mon .idr

“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 __
  is-unital-magma→is-semigroup→is-monoid uni sem .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

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

monoid-inverse-unique {1M = 1M} {__} m e x y li1 ri2 =
  x             ≡⟨ sym (m .idr)
  x ⋆ ⌜ 1M ⌝    ≡˘⟨ ap¡ ri2 ⟩
  x ⋆ (e ⋆ y)   ≡⟨ m .associative ⟩
  ⌜ x ⋆ e ⌝ ⋆ y ≡⟨ ap! li1 ⟩
  1M ⋆ y        ≡⟨ m .idl ⟩
  y             ∎

Constructing monoids🔗

The interface to Monoid-on is contains some annoying nesting, so we provide an interface that arranges the data in a more user-friendly way.

record make-monoid {} (A : Type ℓ) : Type ℓ where
  field
    monoid-is-set : is-set A
    __ : A  A  A
    1M : A
    ⋆-assoc :  x y z  x ⋆ (y ⋆ z)(x ⋆ y) ⋆ z
    ⋆-idl :  x  1M ⋆ x ≡ x
    ⋆-idr :  x  x ⋆ 1M ≡ x

  to-is-monoid : is-monoid 1M __
  to-is-monoid .has-is-semigroup .is-semigroup.has-is-magma = record { has-is-set = monoid-is-set }
  to-is-monoid .has-is-semigroup .is-semigroup.associative = ⋆-assoc _ _ _
  to-is-monoid .idl = ⋆-idl _
  to-is-monoid .idr = ⋆-idr _

  to-monoid-on : Monoid-on A
  to-monoid-on .Monoid-on.identity = 1M
  to-monoid-on .Monoid-on.__ = __
  to-monoid-on .Monoid-on.has-is-monoid = to-is-monoid

open make-monoid using (to-is-monoid; to-monoid-on) public

  1. Counterexample: The map which sends everything to zero is a semigroup homomorphism, but does not preserve the unit of ↩︎