Groups.jl/src/WreathProducts.jl

211 lines
6.1 KiB
Julia
Raw Normal View History

export WreathProduct, WreathProductElem
2017-06-22 14:21:25 +02:00
###############################################################################
#
# WreathProduct / WreathProductElem
#
###############################################################################
2018-09-21 18:08:44 +02:00
@doc doc"""
WreathProduct(N, P) <: Group
> Implements Wreath product of a group `N` by permutation group $P = S_n$,
2017-06-22 14:21:25 +02:00
> usually written as $N \wr P$.
> The multiplication inside wreath product is defined as
> > `(n, σ) * (m, τ) = (n*σ(m), στ)`
> where `σ(m)` denotes the action (from the right) of the permutation group on
> `n-tuples` of elements from `N`
2017-06-22 14:21:25 +02:00
# Arguments:
2019-01-02 15:47:58 +01:00
* `N::Group` : the single factor of the group $N$
* `P::Generic.PermGroup` : full `PermutationGroup`
2017-06-22 14:21:25 +02:00
"""
2019-01-09 16:44:04 +01:00
struct WreathProduct{N, T<:Group, PG<:Generic.PermGroup} <: Group
2019-01-02 15:47:58 +01:00
N::DirectPowerGroup{N, T}
2019-01-09 16:44:04 +01:00
P::PG
2017-06-22 14:21:25 +02:00
2019-01-09 16:44:04 +01:00
function WreathProduct(Gr::T, P::PG) where {T, PG<:Generic.PermGroup}
2019-01-02 10:30:25 +01:00
N = DirectPowerGroup(Gr, Int(P.n))
2019-01-09 16:44:04 +01:00
return new{Int(P.n), T, PG}(N, P)
2017-06-22 14:21:25 +02:00
end
end
2019-01-09 16:44:04 +01:00
struct WreathProductElem{N, T<:GroupElem, P<:Generic.perm} <: GroupElem
2019-01-02 15:47:58 +01:00
n::DirectPowerGroupElem{N, T}
2019-01-09 16:44:04 +01:00
p::P
2017-06-22 14:21:25 +02:00
2019-01-09 16:44:04 +01:00
function WreathProductElem(n::DirectPowerGroupElem{N,T}, p::P,
check::Bool=true) where {N, T, P<:Generic.perm}
if check
2019-01-09 16:44:04 +01:00
N == length(p.d) || throw(DomainError(
2018-07-30 14:59:11 +02:00
"Can't form WreathProductElem: lengths differ"))
end
2019-01-09 16:44:04 +01:00
return new{N, T, P}(n, p)
2017-06-22 14:21:25 +02:00
end
end
###############################################################################
#
# Type and parent object methods
#
###############################################################################
2019-01-09 16:44:04 +01:00
elem_type(::Type{WreathProduct{N, T, PG}}) where {N, T, PG} = WreathProductElem{N, elem_type(T), elem_type(PG)}
2017-06-22 14:21:25 +02:00
2019-01-09 16:44:04 +01:00
parent_type(::Type{WreathProductElem{N, T, P}}) where {N, T, P} =
WreathProduct{N, parent_type(T), parent_type(P)}
2017-06-22 14:21:25 +02:00
parent(g::WreathProductElem) = WreathProduct(parent(g.n[1]), parent(g.p))
2017-06-22 14:21:25 +02:00
###############################################################################
#
# Parent object call overloads
#
###############################################################################
function (G::WreathProduct{N})(g::WreathProductElem{N}) where {N}
n = G.N(g.n)
p = G.P(g.p)
2017-07-21 16:02:07 +02:00
return WreathProductElem(n, p)
2017-06-22 14:21:25 +02:00
end
2018-09-21 18:08:44 +02:00
@doc doc"""
2019-01-02 10:30:25 +01:00
(G::WreathProduct)(n::DirectPowerGroupElem, p::Generic.perm)
2017-06-22 14:21:25 +02:00
> Creates an element of wreath product `G` by coercing `n` and `p` to `G.N` and
> `G.P`, respectively.
"""
2019-01-02 10:30:25 +01:00
(G::WreathProduct)(n::DirectPowerGroupElem, p::Generic.perm) = WreathProductElem(n,p)
2017-06-22 14:21:25 +02:00
(G::WreathProduct)() = WreathProductElem(G.N(), G.P(), false)
2017-06-22 14:21:25 +02:00
2018-09-21 18:08:44 +02:00
@doc doc"""
2017-09-15 18:54:32 +02:00
(G::WreathProduct)(p::Generic.perm)
2017-06-22 14:21:25 +02:00
> Returns the image of permutation `p` in `G` via embedding `p -> (id,p)`.
"""
2017-09-15 18:54:32 +02:00
(G::WreathProduct)(p::Generic.perm) = G(G.N(), p)
2017-06-22 14:21:25 +02:00
2018-09-21 18:08:44 +02:00
@doc doc"""
2019-01-02 10:30:25 +01:00
(G::WreathProduct)(n::DirectPowerGroupElem)
2017-06-22 14:21:25 +02:00
> Returns the image of `n` in `G` via embedding `n -> (n,())`. This is the
> embedding that makes the sequence `1 -> N -> G -> P -> 1` exact.
2017-06-22 14:21:25 +02:00
"""
2019-01-02 10:30:25 +01:00
(G::WreathProduct)(n::DirectPowerGroupElem) = G(n, G.P())
2017-06-22 14:21:25 +02:00
(G::WreathProduct)(n,p) = G(G.N(n), G.P(p))
2017-06-22 14:21:25 +02:00
###############################################################################
#
# Basic manipulation
#
###############################################################################
function hash(G::WreathProduct, h::UInt)
return hash(G.N, hash(G.P, hash(WreathProduct, h)))
end
function hash(g::WreathProductElem, h::UInt)
return hash(g.n, hash(g.p, hash(WreathProductElem, h)))
2017-06-22 14:21:25 +02:00
end
###############################################################################
#
# String I/O
#
###############################################################################
function show(io::IO, G::WreathProduct)
2017-07-21 14:31:05 +02:00
print(io, "Wreath Product of $(G.N.group) by $(G.P)")
2017-06-22 14:21:25 +02:00
end
function show(io::IO, g::WreathProductElem)
print(io, "($(g.n)$(g.p))")
end
###############################################################################
#
# Comparison
#
###############################################################################
function (==)(G::WreathProduct, H::WreathProduct)
G.N == H.N || return false
G.P == H.P || return false
return true
end
function (==)(g::WreathProductElem, h::WreathProductElem)
g.n == h.n || return false
g.p == h.p || return false
return true
end
###############################################################################
#
# Group operations
2017-06-22 14:21:25 +02:00
#
###############################################################################
2019-01-02 10:30:25 +01:00
(p::perm)(n::DirectPowerGroupElem) = DirectPowerGroupElem(n.elts[p.d])
2018-07-30 13:54:15 +02:00
2018-09-21 18:08:44 +02:00
@doc doc"""
2017-06-22 14:21:25 +02:00
*(g::WreathProductElem, h::WreathProductElem)
> Return the wreath product group operation of elements, i.e.
>
2017-07-21 16:05:49 +02:00
> `g*h = (g.n*g.p(h.n), g.p*h.p)`,
2017-06-22 14:21:25 +02:00
>
2017-09-15 18:54:32 +02:00
> where `g.p(h.n)` denotes the action of `g.p::Generic.perm` on
2019-01-02 10:30:25 +01:00
> `h.n::DirectPowerGroupElem` via standard permutation of coordinates.
2017-06-22 14:21:25 +02:00
"""
2017-07-21 15:59:47 +02:00
function *(g::WreathProductElem, h::WreathProductElem)
2018-07-30 13:54:15 +02:00
return WreathProductElem(g.n*g.p(h.n), g.p*h.p, false)
2017-07-21 15:59:47 +02:00
end
2017-06-22 14:21:25 +02:00
2018-09-21 18:47:02 +02:00
^(g::WreathProductElem, n::Integer) = Base.power_by_squaring(g, n)
2018-09-21 18:08:44 +02:00
@doc doc"""
2017-06-22 14:21:25 +02:00
inv(g::WreathProductElem)
> Returns the inverse of element of a wreath product, according to the formula
2017-07-21 16:02:44 +02:00
> `g^-1 = (g.n, g.p)^-1 = (g.p^-1(g.n^-1), g.p^-1)`.
2017-06-22 14:21:25 +02:00
"""
function inv(g::WreathProductElem)
2018-07-30 13:54:15 +02:00
pinv = inv(g.p)
return WreathProductElem(pinv(inv(g.n)), pinv, false)
2017-06-22 14:21:25 +02:00
end
###############################################################################
#
# Misc
#
###############################################################################
matrix_repr(g::WreathProductElem) = Any[matrix_repr(g.p) g.n]
function iterate(G::WreathProduct)
n, state_N = iterate(G.N)
p, state_P = iterate(G.P)
return G(n,p), (state_N, p, state_P)
2017-06-22 14:21:25 +02:00
end
function iterate(G::WreathProduct, state)
state_N, p, state_P = state
res = iterate(G.N, state_N)
if res == nothing
resP = iterate(G.P, state_P)
if resP == nothing
return nothing
else
n, state_N = iterate(G.N)
p, state_P = resP
end
else
n, state_N = res
end
return G(n,p), (state_N, p, state_P)
end
2019-01-09 16:44:04 +01:00
eltype(::Type{WreathProduct{N,G,PG}}) where {N,G,PG} = WreathProductElem{N, elem_type(G), elem_type(PG)}
2017-06-22 14:21:25 +02:00
order(G::WreathProduct) = order(G.P)*order(G.N)
length(G::WreathProduct) = order(G)