1
0
mirror of https://github.com/kalmarek/Groups.jl.git synced 2025-01-24 16:55:28 +01:00
Groups.jl/src/types.jl

268 lines
7.8 KiB
Julia
Raw Normal View History

2021-05-05 01:10:28 +02:00
## "Abstract" definitions
"""
AbstractFPGroup
An Abstract type representing finitely presented groups. Every instance must implement
2021-05-05 01:10:28 +02:00
* `KnuthBendix.alphabet(G::MyFPGroup)`
* `rewriting(G::MyFPGroup)` : return the rewriting object which must implement
> `KnuthBendix.rewrite!(u, v, rewriting(G))`.
E.g. for `G::FreeGroup` `alphabet(G)` is returned, which amounts to free rewriting.
* `ordering(G::MyFPGroup)[ = KnuthBendix.ordering(rewriting(G))]` : return the
(implicit) ordering for the alphabet of `G`.
2021-05-05 02:35:12 +02:00
* `relations(G::MyFPGroup)` : return a set of defining relations.
2021-05-05 01:10:28 +02:00
AbstractFPGroup may also override `word_type(::Type{MyFPGroup}) = Word{UInt8}`,
which controls the word type used for group elements.
If a group has more than `255` generators you need to define e.g.
2021-05-24 01:41:37 +02:00
> `word_type(::Type{MyFPGroup}) = Word{UInt16}`
2021-05-05 01:10:28 +02:00
"""
abstract type AbstractFPGroup <: GroupsCore.Group end
word_type(G::AbstractFPGroup) = word_type(typeof(G))
# the default:
2021-05-16 23:23:16 +02:00
word_type(::Type{<:AbstractFPGroup}) = Word{UInt8}
2021-05-05 01:10:28 +02:00
"""
rewriting(G::AbstractFPGroup)
Return a "rewriting object" for elements of `G`.
The rewriting object must must implement
KnuthBendix.rewrite!(u::AbstractWord, v::AbstractWord, rewriting(G))
For example if `G` is a `FreeGroup` then `alphabet(G)` is returned which results
in free rewriting. For `FPGroup` a rewriting system is returned which may
(or may not) rewrite word `v` to its normal form (depending on e.g. its confluence).
"""
function rewriting end
KnuthBendix.ordering(G::AbstractFPGroup) = ordering(rewriting(G))
KnuthBendix.alphabet(G::AbstractFPGroup) = alphabet(ordering(G))
2021-12-11 13:55:47 +01:00
Base.@propagate_inbounds function (G::AbstractFPGroup)(
word::AbstractVector{<:Integer},
)
@boundscheck @assert all(
l -> 1 <= l <= length(alphabet(G)),
2021-12-11 13:55:47 +01:00
word,
)
2021-05-05 01:10:28 +02:00
return FPGroupElement(word_type(G)(word), G)
end
## Group Interface
Base.one(G::AbstractFPGroup) = FPGroupElement(one(word_type(G)), G)
2021-12-11 13:55:47 +01:00
Base.eltype(::Type{FPG}) where {FPG<:AbstractFPGroup} =
FPGroupElement{FPG,word_type(FPG)}
2021-05-05 01:10:28 +02:00
2021-05-07 18:14:13 +02:00
include("iteration.jl")
2021-05-05 01:10:28 +02:00
GroupsCore.ngens(G::AbstractFPGroup) = length(G.gens)
function GroupsCore.gens(G::AbstractFPGroup, i::Integer)
2021-05-16 23:22:33 +02:00
@boundscheck 1 <= i <= GroupsCore.ngens(G)
2021-05-05 02:33:36 +02:00
l = alphabet(G)[G.gens[i]]
return FPGroupElement(word_type(G)([l]), G)
2021-05-05 01:10:28 +02:00
end
2021-12-11 13:55:47 +01:00
GroupsCore.gens(G::AbstractFPGroup) =
[gens(G, i) for i in 1:GroupsCore.ngens(G)]
2021-05-05 01:10:28 +02:00
# TODO: ProductReplacementAlgorithm
2021-12-11 13:55:47 +01:00
function Base.rand(
rng::Random.AbstractRNG,
rs::Random.SamplerTrivial{<:AbstractFPGroup},
)
2021-05-05 01:10:28 +02:00
l = rand(10:100)
G = rs[]
2021-05-05 02:33:36 +02:00
nletters = length(alphabet(G))
return FPGroupElement(word_type(G)(rand(1:nletters, l)), G)
2021-05-05 01:10:28 +02:00
end
2021-12-11 13:55:47 +01:00
Base.isfinite(::AbstractFPGroup) = (
@warn "using generic isfinite(::AbstractFPGroup): the returned `false` might be wrong"; false
)
2021-05-05 01:10:28 +02:00
## FPGroupElement
2021-06-29 16:52:35 +02:00
abstract type AbstractFPGroupElement{Gr} <: GroupElement end
2021-12-11 13:55:47 +01:00
mutable struct FPGroupElement{Gr<:AbstractFPGroup,W<:AbstractWord} <:
AbstractFPGroupElement{Gr}
2021-05-05 01:10:28 +02:00
word::W
savedhash::UInt
2021-06-29 16:52:35 +02:00
parent::Gr
2021-05-05 01:10:28 +02:00
2021-12-11 13:55:47 +01:00
FPGroupElement(
word::W,
G::AbstractFPGroup,
2022-10-14 01:14:38 +02:00
hash::UInt=UInt(0),
2021-12-11 13:55:47 +01:00
) where {W<:AbstractWord} = new{typeof(G),W}(word, hash, G)
2021-12-11 13:55:47 +01:00
FPGroupElement{Gr,W}(word::AbstractWord, G::Gr) where {Gr,W} =
new{Gr,W}(word, UInt(0), G)
2021-05-05 01:10:28 +02:00
end
2021-12-11 13:55:31 +01:00
Base.show(io::IO, ::Type{<:FPGroupElement{Gr}}) where {Gr} =
print(io, FPGroupElement, "{$Gr, …}")
2021-06-29 16:52:35 +02:00
word(f::AbstractFPGroupElement) = f.word
2021-05-05 01:10:28 +02:00
#convenience
2021-06-29 16:52:35 +02:00
KnuthBendix.alphabet(g::AbstractFPGroupElement) = alphabet(parent(g))
2021-05-05 01:10:28 +02:00
2021-06-29 16:52:35 +02:00
function Base.show(io::IO, f::AbstractFPGroupElement)
2021-05-05 01:10:28 +02:00
f = normalform!(f)
2021-12-11 13:55:47 +01:00
return KnuthBendix.print_repr(io, word(f), alphabet(f))
2021-05-05 01:10:28 +02:00
end
## GroupElement Interface for FPGroupElement
2021-06-29 16:52:35 +02:00
Base.parent(f::AbstractFPGroupElement) = f.parent
2021-05-05 01:10:28 +02:00
2021-06-29 16:52:35 +02:00
function Base.:(==)(g::AbstractFPGroupElement, h::AbstractFPGroupElement)
2021-05-05 01:10:28 +02:00
@boundscheck @assert parent(g) === parent(h)
2021-05-07 18:11:11 +02:00
normalform!(g)
normalform!(h)
2021-05-05 01:10:28 +02:00
hash(g) != hash(h) && return false
return equality_data(g) == equality_data(h)
2021-05-05 01:10:28 +02:00
end
function Base.deepcopy_internal(g::FPGroupElement, stackdict::IdDict)
return FPGroupElement(copy(word(g)), parent(g), g.savedhash)
2021-05-05 01:10:28 +02:00
end
2021-12-11 13:55:47 +01:00
function Base.inv(g::GEl) where {GEl<:AbstractFPGroupElement}
2021-06-29 16:52:35 +02:00
G = parent(g)
return GEl(inv(word(g), alphabet(G)), G)
2021-06-29 16:52:35 +02:00
end
2021-05-05 01:10:28 +02:00
2021-12-11 13:55:47 +01:00
function Base.:(*)(g::GEl, h::GEl) where {GEl<:AbstractFPGroupElement}
2021-05-05 01:10:28 +02:00
@boundscheck @assert parent(g) === parent(h)
2021-06-29 16:52:35 +02:00
return GEl(word(g) * word(h), parent(g))
2021-05-05 01:10:28 +02:00
end
2021-12-11 13:55:47 +01:00
GroupsCore.isfiniteorder(g::AbstractFPGroupElement) =
isone(g) ? true :
(
@warn "using generic isfiniteorder(::AbstractFPGroupElement): the returned `false` might be wrong"; false
)
2021-05-05 01:10:28 +02:00
# additional methods:
2021-06-29 16:52:35 +02:00
Base.isone(g::AbstractFPGroupElement) = (normalform!(g); isempty(word(g)))
2021-05-05 01:10:28 +02:00
## Free Groups
struct FreeGroup{T,O} <: AbstractFPGroup
2021-05-05 01:10:28 +02:00
gens::Vector{T}
ordering::O
2021-05-05 01:10:28 +02:00
function FreeGroup(gens, ordering::KnuthBendix.WordOrdering)
2021-05-05 01:10:28 +02:00
@assert length(gens) == length(unique(gens))
2022-10-13 23:21:42 +02:00
@assert all(l -> l in alphabet(ordering), gens)
return new{eltype(gens),typeof(ordering)}(gens, ordering)
2021-05-05 01:10:28 +02:00
end
end
FreeGroup(gens, A::Alphabet) = FreeGroup(gens, KnuthBendix.LenLex(A))
2021-05-05 02:33:36 +02:00
function FreeGroup(A::Alphabet)
2021-12-11 13:55:47 +01:00
@boundscheck @assert all(
2022-10-13 23:21:42 +02:00
KnuthBendix.hasinverse(l, A) for l in A
2021-12-11 13:55:47 +01:00
)
2022-10-13 23:21:42 +02:00
gens = Vector{eltype(A)}()
invs = Vector{eltype(A)}()
for l in A
l invs && continue
push!(gens, l)
push!(invs, inv(l, A))
end
return FreeGroup(gens, A)
2021-05-05 01:10:28 +02:00
end
2021-05-26 12:12:26 +02:00
function FreeGroup(n::Integer)
symbols = Symbol[]
inverses = Int[]
sizehint!(symbols, 2n)
sizehint!(inverses, 2n)
for i in 1:n
push!(symbols, Symbol(:f, i), Symbol(:F, i))
2021-12-11 13:55:47 +01:00
push!(inverses, 2i, 2i - 1)
2021-05-26 12:12:26 +02:00
end
return FreeGroup(symbols[1:2:2n], Alphabet(symbols, inverses))
end
2021-12-11 13:55:47 +01:00
Base.show(io::IO, F::FreeGroup) =
print(io, "free group on $(ngens(F)) generators")
2021-05-05 01:10:28 +02:00
# mandatory methods:
KnuthBendix.ordering(F::FreeGroup) = F.ordering
rewriting(F::FreeGroup) = alphabet(F) # alphabet(F) = alphabet(ordering(F))
relations(F::FreeGroup) = Pair{eltype(F),eltype(F)}[]
2021-05-05 02:35:12 +02:00
# GroupsCore interface:
# these are mathematically correct
Base.isfinite(::FreeGroup) = false
2021-12-11 13:55:47 +01:00
GroupsCore.isfiniteorder(g::AbstractFPGroupElement{<:FreeGroup}) =
isone(g) ? true : false
2021-05-05 02:35:12 +02:00
## FP Groups
struct FPGroup{T,RW,S} <: AbstractFPGroup
2021-05-05 02:35:12 +02:00
gens::Vector{T}
2021-05-16 23:22:33 +02:00
relations::Vector{Pair{S,S}}
rw::RW
2021-05-05 02:35:12 +02:00
end
relations(G::FPGroup) = G.relations
rewriting(G::FPGroup) = G.rw
2021-05-05 02:35:12 +02:00
function FPGroup(
G::AbstractFPGroup,
2021-05-16 23:22:33 +02:00
rels::AbstractVector{<:Pair{GEl,GEl}};
2022-10-14 01:14:38 +02:00
ordering=KnuthBendix.ordering(G),
kwargs...
2021-05-16 23:22:33 +02:00
) where {GEl<:FPGroupElement}
2021-05-05 02:35:12 +02:00
for (lhs, rhs) in rels
@assert parent(lhs) === parent(rhs) === G
end
2021-05-16 23:22:33 +02:00
word_rels = [word(lhs) => word(rhs) for (lhs, rhs) in [relations(G); rels]]
rws = KnuthBendix.RewritingSystem(word_rels, ordering)
2021-05-05 02:35:12 +02:00
rws = KnuthBendix.knuthbendix(rws, KnuthBendix.Settings(; kwargs...))
2021-05-05 02:35:12 +02:00
return FPGroup(G.gens, rels, KnuthBendix.IndexAutomaton(rws))
2021-05-05 02:35:12 +02:00
end
2021-12-18 19:34:04 +01:00
function Base.show(io::IO, ::MIME"text/plain", G::FPGroup)
print(io, "Finitely presented group generated by:\n\t{")
Base.print_array(io, permutedims(gens(G)))
println(io, " },")
println(io, "subject to relations:")
return Base.print_array(io, relations(G))
end
2021-05-05 02:35:12 +02:00
function Base.show(io::IO, G::FPGroup)
print(io, "")
2021-12-18 19:34:04 +01:00
Base.print_array(io, permutedims(gens(G)))
println(io, " | ")
print(io, "\t ")
Base.print_array(io, permutedims(relations(G)))
return print(io, "")
2021-05-05 02:35:12 +02:00
end
2021-12-11 13:55:31 +01:00
Base.show(io::IO, ::Type{<:FPGroup{T}}) where {T} =
print(io, FPGroup, "{$T, …}")
## GSymbol aka letter of alphabet
abstract type GSymbol end
Base.literal_pow(::typeof(^), t::GSymbol, ::Val{-1}) = inv(t)
2021-06-21 17:53:29 +02:00
function subscriptify(n::Integer)
subscript_0 = Int(0x2080) # Char(0x2080) -> subscript 0
return join([Char(subscript_0 + i) for i in reverse(digits(n))], "")
end