Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/GAP/pkg/deepthought/gap/   (Algebra von RWTH Aachen Version 4.15.1©)  Datei vom 20.5.2025 mit Größe 15 kB image not shown  

Quelle  DeepThought.gd   Sprache: unbekannt

 
#
# DeepThought: This package provides functions for multiplication and other computations in finitely generated nilpotent groups based on the Deep Thought algorithm.
#
# Declarations
#

# QUICK HACK: declare some functions to resolve circular dependencies
# on the long run, clean this up
DeclareGlobalFunction( "DTP_DetermineMultiplicationFunction" );

#! @Chapter The Deep Thought algorithm
#! Polycyclic and, especially, finitely generated nilpotent groups exhibit a rich structure allowing a special approach towards multiplication using polynomials. The so-called Deep Thought algorithm introduced in <Cite Key="LGS"/> computes these polynomials in practice for a suitable presentation of a finitely generated nilpotent group. Such a presentation is of the following form
#!
#! $$ \langle a_1, \ldots, a_n \mid a_i^{s_i} = a_{i+1}^{c_{i, i, i+1}} \cdots a_n^{c_{i, i, n}}, 1 \leq i \leq n, a_j a_i = a_i a_j a_{j+1}^{c_{i, j, j+1}} \cdots a_n^{c_{i, j, n}}, 1 \leq i < j \leq n \rangle $$
#! with $s_i \in \mathbb{N} \cup \{ \infty \}$ and $c_{i, j, k} \in \mathbb{Z}$. If $s_i = \infty$, then the power relation $a_i^{s_i}$ is skipped.
#!<P/>
#! Let $G$ denote the group presented by the above presentation. Then every element in $G$ can be written uniquely in a so-called normal form. That is, if $G_i := \langle a_i, \ldots, a_n \rangle$ and  $r_i := | G_i : G_{i+1}|$, $1 \leq i \leq n$, are the relative orders, then for certain $e_i \in \mathbb{Z}$ each element can be written as
#! $$ a_1^{e_1} \cdots a_n^{e_n} $$
#! with $0 \leq e_i < r_i$ if $r_i < \infty$. A presentation is called confluent if and only if $s_i = r_i$ for all $1 \leq i \leq n$. If a presentation is not confluent, not all functions provided in this package are applicable, see function <C>DTP_DTapplicability</C>. For more theoretical background see for example the documentation of the &GAP; package &Polycyclic;.
#!<P/>
#! The Deep Thought algorithm computes rational polynomials $f_1, \ldots, f_n$ in $2n$ indeterminates such that if $ x := a_1^{x_1} \cdots a_n^{x_n} $ and $y := a_1^{y_1} \cdots a_n^{y_n} $ are two elements (in normal form or not with $x_1, \ldots, x_n, y_1, \ldots, y_n \in \mathbb{Z}$), then their product $xy$ is given by
#! $$a_1^{f_1(x_1, \ldots, x_n, y_1, \ldots, y_n)} \cdots a_n^{f_n(x_1, \ldots, x_n, y_1, \ldots, y_n)}.$$
#! If the collector is confluent, also the normal form of the product can be computed. Otherwise this is not possible.
#! Moreover, there exists a second version of the Deep Thought algorithms which computes $n^2$ polynomials $f_{rs}$, $1 \leq r, s \leq n$, suitable for multiplications of the form
#! $$(a_1^{x_1} \cdots a_n^{x_n}) \cdot a_s^{y_s} = a_1^{f_{1s}(x_1, \ldots, x_n, y_s)} \cdots a_n^{f_{ns}(x_1, \ldots, x_n, y_s)} $$
#! for $1 \leq s \leq n$. Each general multiplication can be expressed using these special multiplications. Depending on the presentation, polynomials of one version may be more efficient for computations than the polynomials of the other version. For a suggestion on which polynomials to use for a given presentation, see <C>DTP_DTapplicability</C>. In the following, Deep Thought type $f_r$ refers to the Deep Thought algorithm which uses $n$ polynomials and type $f_{rs}$ refers to the Deep Thought algorithm using $n^2$ polynomials.
#!<P/>
#! In order to work with the Deep Thought functions, the group presentation is expected to be given as a collector <C>coll</C>, as defined in the &GAP; package &Polycyclic;. Moreover, the &Polycyclic; package introduces the structure of exponent vectors <C>expvec</C>, which will be used also in this package to represent group elements.
#! In the following text, a group element $a_1^{x_1} \cdots a_n^{x_n}$ is identified with its exponent vector in form of the list <C>[x_1, ..., x_n]</C>. For example, if <C>expvec1, expvec2</C> are exponent vectors of elements in the same group, then <C>expvec1 * expvec2</C> describes the multiplication of the corresponding group elements, and so on. Note that generally exponent vectors are not assumed to represent normal forms.
#!<P/>

#! @Section Category DTObj
#! This package uses the category <C>DTObj</C>. A <C>DTObj</C> is a <C>IsFromTheLeftCollectorRep</C> with certain further list entries to store the Deep Thought polynomials of a collector together with some additional information. That is, the functions <C>DTP_DTpols_r</C> and <C>DTP_DTpols_rs</C> return a <C>DTObj</C> which has entries as <C>IsFromTheLeftCollectorRep</C> and additionally:
#! <List>
#! <Item> <C>DTObj![PC_DTPPolynomials]</C>: Deep Thought polynomials in form of (nested) lists</Item>
#! <Item> <C>DTObj![PC_DTPOrders]</C>: list containing orders of group generators if the collector is confluent</Item>
#! <Item> <C>DTObj![PC_DTPConfluent]</C>: boolean value indicating whether the collector is confluent or not</Item>
#! </List>

#! @Chapter Using Deep Thought functions
#! In the following sections, functions provided for computing Deep Thought polynomials and using them for calculations are listed.
#! @Section Computing Deep Thought polynomials

#! @Arguments coll
#! @Returns boolean
#! @Description Checks the collector <C>coll</C> for applicability of Deep Thought functions. Note that depending on confluency some functions may be applicable, while others are not. Information on the applicability and which type of Deep Thought polynomials are suggested is printed to the terminal. Here, "+" means that the following property is fulfilled, otherwise there is a "-". The function returns <C>false</C> if Deep Thought is not applicable to the collector <C>coll</C> and <C>true</C> otherwise. Anyway, even if <C>true</C> is returned, **not all functions need to be applicable** (in case of inconfluencies).
DeclareGlobalFunction( "DTP_DTapplicability" );

#! @Arguments coll, [, rs_flag]
#! @Returns a DTObj
#! @Description Computes a DTObj for the collector coll, either with polynomials of type $f_{rs}$ (if <C>rs_flag = true</C>) or with polynomials of type $f_r$, if <C>rs_flag = false</C>. If the optional argument <C>rs_flag</C> is not provided, polynomials of type $f_{rs}$ are computed. The function checks whether the collector <C>coll</C> is confluent. If not, a warning is displayed. Note that the function assumes the collector <C>coll</C> to be suitable for Deep Thought, see function <C>DTP_DTapplicability</C>.
DeclareGlobalFunction( "DTP_DTObjFromCollector" );

#! @BeginExampleSession
#! gap> G := UnitriangularPcpGroup(10, 0);;
#! gap> coll := Collector(G);;
#! gap> DTP_DTapplicability(coll);
#! Checking collector for DT-applicability. "+" means the following property
#! is fulfilled.
#! +   conjugacy relations
#! +   power relations
#! +   confluent
#! Suggestion: Call DTP_DTObjFromColl with rs_flag = true.
#! true
#! # calling DTP_DTObjFromCollector without rs_flag implies rs_flag = true:
#! gap> DTObj := DTP_DTObjFromCollector(coll);
#! <DTObj>
#! @EndExampleSession

#! @Section Computations with Deep Thought polynomials

#! @Arguments expvec, int, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of <C>expvec</C>$^{int}$. If <C>IsConfluent(DTObj) = true</C>, then the result is in normal form.
DeclareGlobalFunction( "DTP_Exp" );

#! @Arguments expvec, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the inverse of the element corresponding to <C>expvec</C>. If <C>IsConfluent(DTObj) = true</C>, then the result is in normal form.
DeclareGlobalFunction( "DTP_Inverse" );

#! @Arguments expvec, coll
#! @Returns boolean or positive integer
#! @Description Checks whether <C>expvec</C> is in normal form or not. If yes, the return value is <C>true</C>. Otherwise the return value is the smallest generator index for which the normal form condition is violated, i.e. for which the relative order <C>RelativeOrder(coll)[i]</C> is non-zero, and <C>expvec[i]</C> < <C>0</C> or <C>expvec[i]</C> $\geq$ <C>RelativeOrder(coll)[i]</C>.
DeclareGlobalFunction( "DTP_IsInNormalForm" );

#! @Arguments expvec1, expvec2, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the product <C>expvec1 * expvec2</C> using the Deep Thought polynomials. If <C>IsConfluent(DTObj) = true</C>, then the result is returned in normal form.
#! <C>DTP_Multiply</C> either calls <C>DTP_Multiply_r</C> or <C>DTP_Multiply_rs</C> depending on which type of polynomials are stored in <C>DTObj</C>.
DeclareGlobalFunction( "DTP_Multiply" );
#! @Arguments expvec1, expvec2, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the product <C>expvec1 * expvec2</C> using the Deep Thought polynomials of type $f_r$ stored in <C>DT_Obj</C>. If <C>IsConfluent(DTObj) = true</C>, then the result is returned in normal form.
#! @Arguments expvec1, expvec2, DTObj
DeclareGlobalFunction( "DTP_Multiply_r" );
#! @Arguments expvec1, expvec2, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the product <C>expvec1 * expvec2</C> using the Deep Thought polynomials of type $f_{rs}$ stored in <C>DT_Obj</C>. If <C>IsConfluent(DTObj) = true</C>, then the result is returned in normal form.
#! @Arguments expvec1, expvec2, DTObj
DeclareGlobalFunction( "DTP_Multiply_rs" );

#! @Arguments expvec, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the normal form of <C>expvec</C>. For this function to be applicable, we need <C>IsConfluent(DTObj) = true</C>.
DeclareGlobalFunction( "DTP_NormalForm" );

#! @Arguments expvec, DTObj
#! @Returns positive integer or infinity
#! @Description Computes the order of the element described by <C>expvec</C>. For this function to be applicable, we need <C>IsConfluent(DTObj) = true</C>.
DeclareGlobalFunction( "DTP_Order" );

#! @Arguments expvec1, expvec2, DTObj
#! @Returns an exponent vector
#! @Description Computes the exponent vector of the element corresponding to <C>expvec1</C>$^{-1}$ <C>* expvec2</C>, i.e. the result solves the equation <C>expvec1 * result = expvec2</C>. If <C>IsConfluent(DTObj) = true</C>, then the result
#! describes a normal form.
DeclareGlobalFunction( "DTP_SolveEquation" );

#! @BeginExampleSession
#! gap> G := PcGroupToPcpGroup(SmallGroup(23^5, 2));
#! Pcp-group with orders [ 23, 23, 23, 23, 23 ]
#! gap> coll := Collector(G);
#! <<from the left collector with 5 generators>>
#! gap> DTObj := DTP_DTObjFromCollector(coll);
#! <DTObj>
#! gap> g := [100, 134, -31, 52, 5235];
#! [ 100, 134, -31, 52, 5235 ]
#! gap> DTP_IsInNormalForm(g, DTObj);
#! 1
#! gap> g := DTP_NormalForm(g, DTObj);
#! [ 8, 19, 15, 10, 19 ]
#! gap> DTP_IsInNormalForm(g, DTObj);
#! true
#! gap> DTP_Inverse(g, DTObj);
#! [ 15, 4, 22, 12, 3 ]
#! gap> DTP_Order(g, DTObj);
#! 529
#! gap> h := [142, 2, -41, 23, 1];
#! [ 142, 2, -41, 23, 1 ]
#! gap> DTP_Multiply(g, h, DTObj);
#! [ 12, 21, 4, 16, 20 ]
#! @EndExampleSession

#! @Section Computations with pcp-elements
#! When Deep Thought polynomials are available, certain computations allow different approaches which may be faster than the methods used by default.
#! In this section, computations for which such extra functions taking pcp-elements as input are available are listed. All of these functions expect the collector belonging to the pcp-elements to be a <C>DTObj</C>.

#! @Arguments pcp-element, int
#! @Returns pcp-element
#! @Description Returns the pcp-element <C>pcp-element</C>$^{int}$. If <C>IsConfluent(DTObj) = true</C>, then the result is in normal form.
DeclareGlobalFunction( "DTP_PCP_Exp" );

#! @Arguments pcp-element
#! @Returns pcp-element
#! @Description Returns the pcp-elment <C>pcp-element^-1</C>. If <C>IsConfluent(DTObj) = true</C>, then the result is in normal form.
DeclareGlobalFunction( "DTP_PCP_Inverse" );

#! @Arguments pcp-element
#! @Returns pcp-element
#! @Description Returns a pcp-element which is the normal form of the input pcp-element. For this function to be applicable, we need <C>IsConfluent(DTObj) = true</C>.
DeclareGlobalFunction( "DTP_PCP_NormalForm" );

#! @Arguments pcp-element
#! @Returns positive integer or infinity
#! @Description Computes the order of the pcp-element. For this function to be applicable, we need <C>IsConfluent(DTObj) = true</C>.
DeclareGlobalFunction( "DTP_PCP_Order" );

#! @Arguments pcp-element1, pcp-element2
#! @Returns pcp-element
#! @Description Returns the pcp-element <C>pcp-element1</C>$^{-1}$ <C>* pcp-element2</C>, i.e. the result solves the equation <C>pcp-element1 * pcp-element = pcp-element2</C>. If <C>IsConfluent(DTObj) = true</C>, then the result
#! describes a normal form.
DeclareGlobalFunction( "DTP_PCP_SolveEquation" );

#! @BeginExampleSession
#! gap> G := HeisenbergPcpGroup(7);;
#! gap> coll := Collector(G);;
#! gap> DTObj := DTP_DTObjFromCollector(coll);;
#! gap> H := PcpGroupByCollector(DTObj);;
#! gap> g := Random(H);; h := Random(H);;
#! gap> DTP_PCP_SolveEquation(g, h);
#! g1^-3*g2^-1*g3^-7*g4*g5^-6*g6*g7*g8^2*g9^3*g11^-4*g12^5*g14^-2*g15^7
# g^-1 * h;
#! g1^-3*g2^-1*g3^-7*g4*g5^-6*g6*g7*g8^2*g9^3*g11^-4*g12^5*g14^-2*g15^7
# Order(g);
#! infinity
#! gap> g^-1;
#! g1^-2*g3^-3*g4^-1*g5^-4*g6^2*g7*g8^-3*g10^-3*g11^-1*g12^4*g14^-2*g15^-3
#! gap> DTP_PCP_Inverse(h);
#! g1*g2*g3^4*g4^-2*g5^2*g6*g8^-5*g9^-3*g10^-3*g11^3*g12^-1*g15^-33
#! @EndExampleSession

#! @Section Accessing Deep Thought polynomials
#!  In this section, functions which can be used to display the content of a <C>DTObj</C> are documented. Furthermore, Deep Thought polynomials stored in a <C>DTObj</C> can be converted to &GAP; polynomials.

#! @Arguments DTObj
#! @Returns nothing
#! @Description Prints information about <C>DTObj</C> to the terminal. In particular, the Deep Thought polynomials are printed in human-readable form. This function is also called by the method of <C>Display</C> for a <C>DTObj</C>.
DeclareGlobalFunction( "DTP_Display_DTObj" );

#! @Arguments DTObj
#! @Returns list
#! @Description Converts the Deep Thought polynomials stored in <C>DTObj[PC_DTPPolynomials]</C> to &GAP; polynomials and returns them in a list together with their polynomial ring.
DeclareGlobalFunction( "DTP_pols2GAPpols" );

#! @BeginExampleSession
#! gap> coll := FromTheLeftCollector(4);;
#! gap> SetConjugate(coll, 2, 1, [2, 1, 3, 2]);
#! gap> SetConjugate(coll, 3, 1, [3, 1, 4, 1]);
#! gap> SetConjugate(coll, 3, 2, [3, 1, 4, 5]);
#! gap> UpdatePolycyclicCollector(coll);
#! gap> DTObj := DTP_DTObjFromCollector(coll);
#! <DTObj>
#! gap> Display(DTObj);
#! Polynomials f_rs for s = 1:
#! f_1,s = X_1 + Y_1
#! f_2,s = X_2
#! f_3,s = X_3 + 2 * X_2 Y_1
#! f_4,s = X_4 + X_3 Y_1 + 2 * X_2 Binomial(Y_1, 2) + 10 * Binomial(X_2, 2) Y_1
#! Polynomials f_rs for s = 2:
#! f_1,s = X_1
#! f_2,s = X_2 + Y_2
#! f_3,s = X_3
#! f_4,s = X_4 + 5 * X_3 Y_2
#! Polynomials f_rs for s = 3:
#! f_1,s = X_1
#! f_2,s = X_2
#! f_3,s = X_3 + Y_3
#! f_4,s = X_4
#! Polynomials f_rs for s = 4:
#! f_1,s = X_1
#! f_2,s = X_2
#! f_3,s = X_3
#! f_4,s = X_4 + Y_4
#! gap> DTObj := DTP_DTObjFromCollector(coll, false);
#! <DTObj>
#! gap> Display(DTObj);
#! f_1 = X_1 + Y_1
#! f_2 = X_2 + Y_2
#! f_3 = X_3 + Y_3 + 2 * X_2 Y_1
#! f_4 = X_4 + Y_4 + X_3 Y_1 + 2 * X_2 Binomial(Y_1, 2) +
#! 10 * Binomial(X_2, 2) Y_1 + 5 * X_3 Y_2 + 10 * X_2 Y_1 Y_2
#! gap> DTP_pols2GAPpols(DTObj);
#!  [ [ x1+y1, x2+y2, 2*x2*y1+x3+y3,
#! 5*x2^2*y1+x2*y1^2+10*x2*y1*y2-6*x2*y1+x3*y1+5*x3*y2+x4+y4 ],
#! Rationals[x1,x2,x3,x4,y1,y2,y3,y4] ]
#! @EndExampleSession

[ Dauer der Verarbeitung: 0.26 Sekunden  (vorverarbeitet)  ]