Quadratic forms overview¶
AUTHORS:
- Jon Hanke (2007-06-19) 
- Anna Haensch (2010-07-01): Formatting and ReSTification 
- Simon Brandhorst (2019-10-15): - quadratic_form_from_invariants()
- sage.quadratic_forms.quadratic_form.DiagonalQuadraticForm(R, diag)[source]¶
- Return a quadratic form over \(R\) which is a sum of squares. - INPUT: - R– ring
- diag– list/tuple of elements coercible to \(R\)
 - OUTPUT: quadratic form - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 5 0 ] [ * * * 7 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 5 0 ] [ * * * 7 ] 
- class sage.quadratic_forms.quadratic_form.QuadraticForm(R, n=None, entries=None, unsafe_initialization=False, number_of_automorphisms=None, determinant=None)[source]¶
- Bases: - SageObject- The - QuadraticFormclass represents a quadratic form in \(n\) variables with coefficients in the ring \(R\).- INPUT: - The constructor may be called in any of the following ways. - QuadraticForm(R, n, entries), where- R– ring for which the quadratic form is defined
- n– integer \(\geq 0\)
- entries– list of \(n(n+1)/2\) coefficients of the quadratic form in \(R\) (given lexicographically, or equivalently, by rows of the matrix)
 
- QuadraticForm(p), where- p– a homogeneous polynomial of degree \(2\)
 
- QuadraticForm(R, n), where- R– a ring
- n– a symmetric \(n \times n\) matrix with even diagonal (relative to \(R\))
 
- QuadraticForm(R), where- R– a symmetric \(n \times n\) matrix with even diagonal (relative to its base ring)
 
 - If the keyword argument - unsafe_initializeis True, then the subsequent fields may by used to force the external initialization of various fields of the quadratic form. Currently the only fields which can be set are:- number_of_automorphisms
- determinant
 - OUTPUT: quadratic form - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [1,2,3,4,5,6]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1),Integer(2),Integer(3),Integer(4),Integer(5),Integer(6)]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] - sage: Q = QuadraticForm(QQ, 3, [1,2,3,4/3,5,6]); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4/3 5 ] [ * * 6 ] sage: Q[0,0] 1 sage: Q[0,0].parent() Rational Field - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(3), [Integer(1),Integer(2),Integer(3),Integer(4)/Integer(3),Integer(5),Integer(6)]); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4/3 5 ] [ * * 6 ] >>> Q[Integer(0),Integer(0)] 1 >>> Q[Integer(0),Integer(0)].parent() Rational Field - sage: Q = QuadraticForm(QQ, 7, range(28)); Q Quadratic form in 7 variables over Rational Field with coefficients: [ 0 1 2 3 4 5 6 ] [ * 7 8 9 10 11 12 ] [ * * 13 14 15 16 17 ] [ * * * 18 19 20 21 ] [ * * * * 22 23 24 ] [ * * * * * 25 26 ] [ * * * * * * 27 ] - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(7), range(Integer(28))); Q Quadratic form in 7 variables over Rational Field with coefficients: [ 0 1 2 3 4 5 6 ] [ * 7 8 9 10 11 12 ] [ * * 13 14 15 16 17 ] [ * * * 18 19 20 21 ] [ * * * * 22 23 24 ] [ * * * * * 25 26 ] [ * * * * * * 27 ] - sage: Q = QuadraticForm(QQ, 2, range(1,4)) sage: A = Matrix(ZZ, 2, 2, [-1,0,0,1]) sage: Q(A) Quadratic form in 2 variables over Rational Field with coefficients: [ 1 -2 ] [ * 3 ] - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(2), range(Integer(1),Integer(4))) >>> A = Matrix(ZZ, Integer(2), Integer(2), [-Integer(1),Integer(0),Integer(0),Integer(1)]) >>> Q(A) Quadratic form in 2 variables over Rational Field with coefficients: [ 1 -2 ] [ * 3 ] - sage: m = matrix(2, 2, [1,2,3,4]) sage: m + m.transpose() [2 5] [5 8] sage: QuadraticForm(m + m.transpose()) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 5 ] [ * 4 ] - >>> from sage.all import * >>> m = matrix(Integer(2), Integer(2), [Integer(1),Integer(2),Integer(3),Integer(4)]) >>> m + m.transpose() [2 5] [5 8] >>> QuadraticForm(m + m.transpose()) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 5 ] [ * 4 ] - sage: P.<x,y,z> = QQ[] sage: p = x^2 + 2*x*y + x*z/2 + y^2 + y*z/3 sage: QuadraticForm(p) Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 1/2 ] [ * 1 1/3 ] [ * * 0 ] - >>> from sage.all import * >>> P = QQ['x, y, z']; (x, y, z,) = P._first_ngens(3) >>> p = x**Integer(2) + Integer(2)*x*y + x*z/Integer(2) + y**Integer(2) + y*z/Integer(3) >>> QuadraticForm(p) Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 1/2 ] [ * 1 1/3 ] [ * * 0 ] - sage: QuadraticForm(ZZ, m + m.transpose()) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 5 ] [ * 4 ] - >>> from sage.all import * >>> QuadraticForm(ZZ, m + m.transpose()) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 5 ] [ * 4 ] - sage: QuadraticForm(QQ, m + m.transpose()) Quadratic form in 2 variables over Rational Field with coefficients: [ 1 5 ] [ * 4 ] - >>> from sage.all import * >>> QuadraticForm(QQ, m + m.transpose()) Quadratic form in 2 variables over Rational Field with coefficients: [ 1 5 ] [ * 4 ] - CS_genus_symbol_list(force_recomputation=False)[source]¶
- Return the list of Conway-Sloane genus symbols in increasing order of primes dividing 2*det. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3,4]) sage: Q.CS_genus_symbol_list() [Genus symbol at 2: [2^-2 4^1 8^1]_6, Genus symbol at 3: 1^3 3^-1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3),Integer(4)]) >>> Q.CS_genus_symbol_list() [Genus symbol at 2: [2^-2 4^1 8^1]_6, Genus symbol at 3: 1^3 3^-1] 
 - GHY_mass__maximal()[source]¶
- Use the GHY formula to compute the mass of a (maximal?) quadratic lattice. This works for any number field. - REFERENCES: - See [GHY, Prop 7.4 and 7.5, p121] and [GY, Thrm 10.20, p25]. - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.GHY_mass__maximal() - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.GHY_mass__maximal() 
 - Gram_det()[source]¶
- Return the determinant of the Gram matrix of \(Q\). - Note - This is defined over the fraction field of the ring of the quadratic form, but is often not defined over the same ring as the quadratic form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.Gram_det() 2 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.Gram_det() 2 
 - Gram_matrix()[source]¶
- Return a (symmetric) Gram matrix \(A\) for the quadratic form \(Q\), meaning that \[Q(x) = x^t\cdot A\cdot x,\]- defined over the base ring of \(Q\). If this is not possible, then a - TypeErroris raised.- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: A = Q.Gram_matrix(); A [1 0 0 0] [0 3 0 0] [0 0 5 0] [0 0 0 7] sage: A.base_ring() Integer Ring - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> A = Q.Gram_matrix(); A [1 0 0 0] [0 3 0 0] [0 0 5 0] [0 0 0 7] >>> A.base_ring() Integer Ring 
 - Gram_matrix_rational()[source]¶
- Return a (symmetric) Gram matrix \(A\) for the quadratic form \(Q\), meaning that \[Q(x) = x^t\cdot A\cdot x,\]- defined over the fraction field of the base ring. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: A = Q.Gram_matrix_rational(); A [1 0 0 0] [0 3 0 0] [0 0 5 0] [0 0 0 7] sage: A.base_ring() Rational Field - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> A = Q.Gram_matrix_rational(); A [1 0 0 0] [0 3 0 0] [0 0 5 0] [0 0 0 7] >>> A.base_ring() Rational Field 
 - Hessian_matrix()[source]¶
- Return the Hessian matrix \(A\) for which \(Q(X) = (1/2) X^t\cdot A\cdot X\). - EXAMPLES: - sage: Q = QuadraticForm(QQ, 2, range(1,4)); Q Quadratic form in 2 variables over Rational Field with coefficients: [ 1 2 ] [ * 3 ] sage: Q.Hessian_matrix() [2 2] [2 6] sage: Q.matrix().base_ring() Rational Field - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(2), range(Integer(1),Integer(4))); Q Quadratic form in 2 variables over Rational Field with coefficients: [ 1 2 ] [ * 3 ] >>> Q.Hessian_matrix() [2 2] [2 6] >>> Q.matrix().base_ring() Rational Field 
 - Kitaoka_mass_at_2()[source]¶
- Return the local mass of the quadratic form when \(p=2\), according to Theorem 5.6.3 on pp108–9 of Kitaoka’s Book “The Arithmetic of Quadratic Forms”. - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.Kitaoka_mass_at_2() # WARNING: WE NEED TO CHECK THIS CAREFULLY! 1/2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.Kitaoka_mass_at_2() # WARNING: WE NEED TO CHECK THIS CAREFULLY! 1/2 
 - Pall_mass_density_at_odd_prime(p)[source]¶
- Return the local representation density of a form (for representing itself) defined over \(\ZZ\), at some prime \(p>2\). - REFERENCES: - Pall’s article “The Weight of a Genus of Positive n-ary Quadratic Forms” appearing in Proc. Symp. Pure Math. VIII (1965), pp95–105. - INPUT: - p– a prime number > 2
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [1,0,0,1,0,1]) sage: Q.Pall_mass_density_at_odd_prime(3) [(0, Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 1 0 ] [ * * 1 ])] [(0, 3, 8)] [8/9] 8/9 8/9 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1),Integer(0),Integer(0),Integer(1),Integer(0),Integer(1)]) >>> Q.Pall_mass_density_at_odd_prime(Integer(3)) [(0, Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 1 0 ] [ * * 1 ])] [(0, 3, 8)] [8/9] 8/9 8/9 
 - Watson_mass_at_2()[source]¶
- Return the local mass of the quadratic form when \(p=2\), according to Watson’s Theorem 1 of “The 2-adic density of a quadratic form” in Mathematika 23 (1976), pp 94–106. - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.Watson_mass_at_2() # WARNING: WE NEED TO CHECK THIS CAREFULLY! # needs sage.symbolic 384 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.Watson_mass_at_2() # WARNING: WE NEED TO CHECK THIS CAREFULLY! # needs sage.symbolic 384 
 - add_symmetric(c, i, j, in_place=False)[source]¶
- Perform the substitution \(x_j \longmapsto x_j + c\cdot x_i\), which has the effect (on associated matrices) of symmetrically adding \(c\) times the \(j\)-th row/column to the \(i\)-th row/column. - NOTE: This is meant for compatibility with previous code, which implemented a matrix model for this class. It is used in the method - local_normal_form().- INPUT: - c– an element of- self.base_ring()
- i,- j– integers \(\geq 0\)
 - OUTPUT: - a - QuadraticForm(by default, otherwise none)- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, range(1,7)); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] sage: Q.add_symmetric(-1, 1, 0) Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 3 ] [ * 3 2 ] [ * * 6 ] sage: Q.add_symmetric(-3/2, 2, 0) # ERROR: -3/2 isn't in the base ring ZZ Traceback (most recent call last): ... RuntimeError: this coefficient cannot be coerced to an element of the base ring for the quadratic form - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), range(Integer(1),Integer(7))); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] >>> Q.add_symmetric(-Integer(1), Integer(1), Integer(0)) Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 3 ] [ * 3 2 ] [ * * 6 ] >>> Q.add_symmetric(-Integer(3)/Integer(2), Integer(2), Integer(0)) # ERROR: -3/2 isn't in the base ring ZZ Traceback (most recent call last): ... RuntimeError: this coefficient cannot be coerced to an element of the base ring for the quadratic form - sage: Q = QuadraticForm(QQ, 3, range(1,7)); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] sage: Q.add_symmetric(-3/2, 2, 0) Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 0 ] [ * 4 2 ] [ * * 15/4 ] - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(3), range(Integer(1),Integer(7))); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] >>> Q.add_symmetric(-Integer(3)/Integer(2), Integer(2), Integer(0)) Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 0 ] [ * 4 2 ] [ * * 15/4 ] 
 - adjoint()[source]¶
- This gives the adjoint (integral) quadratic form associated to the given form, essentially defined by taking the adjoint of the matrix. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,5]) sage: Q.adjoint() Quadratic form in 2 variables over Integer Ring with coefficients: [ 5 -2 ] [ * 1 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(5)]) >>> Q.adjoint() Quadratic form in 2 variables over Integer Ring with coefficients: [ 5 -2 ] [ * 1 ] - sage: Q = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q.adjoint() Quadratic form in 3 variables over Integer Ring with coefficients: [ 39 2 8 ] [ * 19 4 ] [ * * 8 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q.adjoint() Quadratic form in 3 variables over Integer Ring with coefficients: [ 39 2 8 ] [ * 19 4 ] [ * * 8 ] 
 - adjoint_primitive()[source]¶
- Return the primitive adjoint of the quadratic form, which is the smallest discriminant integer-valued quadratic form whose matrix is a scalar multiple of the inverse of the matrix of the given quadratic form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.adjoint_primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 3 -2 ] [ * 1 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.adjoint_primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 3 -2 ] [ * 1 ] 
 - anisotropic_primes()[source]¶
- Return a list with all of the anisotropic primes of the quadratic form. - The infinite place is denoted by \(-1\). - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.anisotropic_primes() # needs sage.libs.pari [2, -1] sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.anisotropic_primes() # needs sage.libs.pari [2, -1] sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,1]) sage: Q.anisotropic_primes() # needs sage.libs.pari [-1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.anisotropic_primes() # needs sage.libs.pari [2, -1] >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.anisotropic_primes() # needs sage.libs.pari [2, -1] >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.anisotropic_primes() # needs sage.libs.pari [-1] 
 - antiadjoint()[source]¶
- This gives an (integral) form such that its adjoint is the given form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q.adjoint().antiadjoint() Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 -1 ] [ * 2 -1 ] [ * * 5 ] sage: Q.antiadjoint() # needs sage.symbolic Traceback (most recent call last): ... ValueError: not an adjoint - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q.adjoint().antiadjoint() Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 -1 ] [ * 2 -1 ] [ * * 5 ] >>> Q.antiadjoint() # needs sage.symbolic Traceback (most recent call last): ... ValueError: not an adjoint 
 - automorphism_group()[source]¶
- Return the group of automorphisms of the quadratic form. - OUTPUT: a - MatrixGroup- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.automorphism_group() Matrix group over Rational Field with 3 generators ( [ 0 0 1] [1 0 0] [ 1 0 0] [-1 0 0] [0 0 1] [ 0 -1 0] [ 0 1 0], [0 1 0], [ 0 0 1] ) - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.automorphism_group() Matrix group over Rational Field with 3 generators ( [ 0 0 1] [1 0 0] [ 1 0 0] [-1 0 0] [0 0 1] [ 0 -1 0] [ 0 1 0], [0 1 0], [ 0 0 1] ) - sage: DiagonalQuadraticForm(ZZ, [1,3,5,7]).automorphism_group() Matrix group over Rational Field with 4 generators ( [-1 0 0 0] [ 1 0 0 0] [ 1 0 0 0] [ 1 0 0 0] [ 0 -1 0 0] [ 0 -1 0 0] [ 0 1 0 0] [ 0 1 0 0] [ 0 0 -1 0] [ 0 0 1 0] [ 0 0 -1 0] [ 0 0 1 0] [ 0 0 0 -1], [ 0 0 0 1], [ 0 0 0 1], [ 0 0 0 -1] ) - >>> from sage.all import * >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]).automorphism_group() Matrix group over Rational Field with 4 generators ( [-1 0 0 0] [ 1 0 0 0] [ 1 0 0 0] [ 1 0 0 0] [ 0 -1 0 0] [ 0 -1 0 0] [ 0 1 0 0] [ 0 1 0 0] [ 0 0 -1 0] [ 0 0 1 0] [ 0 0 -1 0] [ 0 0 1 0] [ 0 0 0 -1], [ 0 0 0 1], [ 0 0 0 1], [ 0 0 0 -1] ) - The smallest possible automorphism group has order two, since we can always change all signs: - sage: Q = QuadraticForm(ZZ, 3, [2, 1, 2, 2, 1, 3]) sage: Q.automorphism_group() Matrix group over Rational Field with 1 generators ( [-1 0 0] [ 0 -1 0] [ 0 0 -1] ) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(2), Integer(1), Integer(2), Integer(2), Integer(1), Integer(3)]) >>> Q.automorphism_group() Matrix group over Rational Field with 1 generators ( [-1 0 0] [ 0 -1 0] [ 0 0 -1] ) 
 - automorphisms()[source]¶
- Return the list of the automorphisms of the quadratic form. - OUTPUT: list of matrices - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.number_of_automorphisms() 48 sage: 2^3 * factorial(3) 48 sage: len(Q.automorphisms()) # needs sage.libs.gap 48 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.number_of_automorphisms() 48 >>> Integer(2)**Integer(3) * factorial(Integer(3)) 48 >>> len(Q.automorphisms()) # needs sage.libs.gap 48 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.number_of_automorphisms() 16 sage: aut = Q.automorphisms() # needs sage.libs.gap sage: len(aut) # needs sage.libs.gap 16 sage: all(Q(M) == Q for M in aut) # needs sage.libs.gap True sage: Q = QuadraticForm(ZZ, 3, [2, 1, 2, 2, 1, 3]) sage: sorted(Q.automorphisms()) # needs sage.libs.gap [ [-1 0 0] [1 0 0] [ 0 -1 0] [0 1 0] [ 0 0 -1], [0 0 1] ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.number_of_automorphisms() 16 >>> aut = Q.automorphisms() # needs sage.libs.gap >>> len(aut) # needs sage.libs.gap 16 >>> all(Q(M) == Q for M in aut) # needs sage.libs.gap True >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(2), Integer(1), Integer(2), Integer(2), Integer(1), Integer(3)]) >>> sorted(Q.automorphisms()) # needs sage.libs.gap [ [-1 0 0] [1 0 0] [ 0 -1 0] [0 1 0] [ 0 0 -1], [0 0 1] ] 
 - base_change_to(*args, **kwds)[source]¶
- Deprecated: Use - change_ring()instead. See Issue #35248 for details.
 - base_ring()[source]¶
- Return the ring over which the quadratic form is defined. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.base_ring() Integer Ring - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.base_ring() Integer Ring 
 - basiclemma(M)[source]¶
- Find a number represented by - selfand coprime to \(M\).- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [2, 1, 3]) sage: Q.basiclemma(6) 71 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2), Integer(1), Integer(3)]) >>> Q.basiclemma(Integer(6)) 71 
 - basiclemmavec(M)[source]¶
- Find a vector where the value of the quadratic form is coprime to \(M\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [2, 1, 5]) sage: Q.basiclemmavec(10) (6, 5) sage: Q(_) 227 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2), Integer(1), Integer(5)]) >>> Q.basiclemmavec(Integer(10)) (6, 5) >>> Q(_) 227 
 - basis_of_short_vectors(show_lengths=False)[source]¶
- Return a basis for \(\ZZ^n\) made of vectors with minimal lengths \(Q(v)\). - OUTPUT: a tuple of vectors, and optionally a tuple of values for each vector - This uses pari:qfminim. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.basis_of_short_vectors() ((1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1)) sage: Q.basis_of_short_vectors(True) (((1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1)), (1, 3, 5, 7)) - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.basis_of_short_vectors() ((1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1)) >>> Q.basis_of_short_vectors(True) (((1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1)), (1, 3, 5, 7)) - The returned vectors are immutable: - sage: v = Q.basis_of_short_vectors()[0] sage: v (1, 0, 0, 0) sage: v[0] = 0 Traceback (most recent call last): ... ValueError: vector is immutable; please change a copy instead (use copy()) - >>> from sage.all import * >>> v = Q.basis_of_short_vectors()[Integer(0)] >>> v (1, 0, 0, 0) >>> v[Integer(0)] = Integer(0) Traceback (most recent call last): ... ValueError: vector is immutable; please change a copy instead (use copy()) 
 - bilinear_map(v, w)[source]¶
- Return the value of the associated bilinear map on two vectors. - Given a quadratic form \(Q\) over some base ring \(R\) with characteristic not equal to 2, this gives the image of two vectors with coefficients in \(R\) under the associated bilinear map \(B\), given by the relation \(2 B(v,w) = Q(v) + Q(w) - Q(v+w)\). - INPUT: - v,- w– two vectors
 - OUTPUT: an element of the base ring \(R\) - EXAMPLES: - First, an example over \(\ZZ\): - sage: Q = QuadraticForm(ZZ, 3, [1,4,0,1,4,1]) sage: v = vector(ZZ, (1,2,0)) sage: w = vector(ZZ, (0,1,1)) sage: Q.bilinear_map(v, w) 8 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1),Integer(4),Integer(0),Integer(1),Integer(4),Integer(1)]) >>> v = vector(ZZ, (Integer(1),Integer(2),Integer(0))) >>> w = vector(ZZ, (Integer(0),Integer(1),Integer(1))) >>> Q.bilinear_map(v, w) 8 - This also works over \(\QQ\): - sage: Q = QuadraticForm(QQ, 2, [1/2,2,1]) sage: v = vector(QQ, (1,1)) sage: w = vector(QQ, (1/2,2)) sage: Q.bilinear_map(v, w) 19/4 - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(2), [Integer(1)/Integer(2),Integer(2),Integer(1)]) >>> v = vector(QQ, (Integer(1),Integer(1))) >>> w = vector(QQ, (Integer(1)/Integer(2),Integer(2))) >>> Q.bilinear_map(v, w) 19/4 - The vectors must have the correct length: - sage: Q = DiagonalQuadraticForm(ZZ, [1,7,7]) sage: v = vector((1,2)) sage: w = vector((1,1,1)) sage: Q.bilinear_map(v, w) Traceback (most recent call last): ... TypeError: vectors must have length 3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(7),Integer(7)]) >>> v = vector((Integer(1),Integer(2))) >>> w = vector((Integer(1),Integer(1),Integer(1))) >>> Q.bilinear_map(v, w) Traceback (most recent call last): ... TypeError: vectors must have length 3 - This does not work if the characteristic is 2: - sage: # needs sage.rings.finite_rings sage: Q = DiagonalQuadraticForm(GF(2), [1,1,1]) sage: v = vector((1,1,1)) sage: w = vector((1,1,1)) sage: Q.bilinear_map(v, w) Traceback (most recent call last): ... TypeError: not defined for rings of characteristic 2 - >>> from sage.all import * >>> # needs sage.rings.finite_rings >>> Q = DiagonalQuadraticForm(GF(Integer(2)), [Integer(1),Integer(1),Integer(1)]) >>> v = vector((Integer(1),Integer(1),Integer(1))) >>> w = vector((Integer(1),Integer(1),Integer(1))) >>> Q.bilinear_map(v, w) Traceback (most recent call last): ... TypeError: not defined for rings of characteristic 2 
 - change_ring(R)[source]¶
- Alters the quadratic form to have all coefficients defined over the new base ring \(R\). Here \(R\) must be coercible to from the current base ring. - Note - This is preferable to performing an explicit coercion through the - base_ring()method, which does not affect the individual coefficients. This is particularly useful for performing fast modular arithmetic evaluations.- INPUT: - R– a ring
 - OUTPUT: quadratic form - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 1 ] sage: Q1 = Q.change_ring(IntegerModRing(5)); Q1 Quadratic form in 2 variables over Ring of integers modulo 5 with coefficients: [ 1 0 ] [ * 1 ] sage: Q1([35,11]) 1 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 1 ] >>> Q1 = Q.change_ring(IntegerModRing(Integer(5))); Q1 Quadratic form in 2 variables over Ring of integers modulo 5 with coefficients: [ 1 0 ] [ * 1 ] >>> Q1([Integer(35),Integer(11)]) 1 
 - cholesky_decomposition(bit_prec=53)[source]¶
- Give the Cholesky decomposition of this quadratic form \(Q\) as a real matrix of precision - bit_prec.- RESTRICTIONS: - \(Q\) must be given as a - QuadraticFormdefined over \(\ZZ\), \(\QQ\), or some real field. If it is over some real field, then an error is raised if the precision given is not less than the defined precision of the real field defining the quadratic form!- REFERENCE: - Cohen’s “A Course in Computational Algebraic Number Theory” book, p 103. 
 - INPUT: - bit_prec– a natural number (default: 53)
 - OUTPUT: an upper triangular real matrix of precision - bit_prec- Todo - If we only care about working over the real double field ( - RDF), then we can use the method- cholesky()present for square matrices over that.- Note - There is a note in the original code reading - Finds the Cholesky decomposition of a quadratic form -- as an upper-triangular matrix! (It's assumed to be global, hence twice the form it refers to.) <-- Python revision asks: Is this true?!? =| - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.cholesky_decomposition() [ 1.00000000000000 0.000000000000000 0.000000000000000] [0.000000000000000 1.00000000000000 0.000000000000000] [0.000000000000000 0.000000000000000 1.00000000000000] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.cholesky_decomposition() [ 1.00000000000000 0.000000000000000 0.000000000000000] [0.000000000000000 1.00000000000000 0.000000000000000] [0.000000000000000 0.000000000000000 1.00000000000000] - sage: Q = QuadraticForm(QQ, 3, range(1,7)); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] sage: Q.cholesky_decomposition() [ 1.00000000000000 1.00000000000000 1.50000000000000] [0.000000000000000 3.00000000000000 0.333333333333333] [0.000000000000000 0.000000000000000 3.41666666666667] - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(3), range(Integer(1),Integer(7))); Q Quadratic form in 3 variables over Rational Field with coefficients: [ 1 2 3 ] [ * 4 5 ] [ * * 6 ] >>> Q.cholesky_decomposition() [ 1.00000000000000 1.00000000000000 1.50000000000000] [0.000000000000000 3.00000000000000 0.333333333333333] [0.000000000000000 0.000000000000000 3.41666666666667] 
 - clifford_conductor()[source]¶
- Return the product of all primes where the Clifford invariant is \(-1\). - Note - For ternary forms, this is the discriminant of the quaternion algebra associated to the quadratic space (i.e. the even Clifford algebra). - EXAMPLES: - sage: # needs sage.libs.pari sage: Q = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q.clifford_invariant(2) 1 sage: Q.clifford_invariant(37) -1 sage: Q.clifford_conductor() 37 sage: DiagonalQuadraticForm(ZZ, [1, 1, 1]).clifford_conductor() # needs sage.libs.pari 2 sage: QuadraticForm(ZZ, 3, [2, -2, 0, 2, 0, 5]).clifford_conductor() # needs sage.libs.pari 30 - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q.clifford_invariant(Integer(2)) 1 >>> Q.clifford_invariant(Integer(37)) -1 >>> Q.clifford_conductor() 37 >>> DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1), Integer(1)]).clifford_conductor() # needs sage.libs.pari 2 >>> QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(2), Integer(0), Integer(2), Integer(0), Integer(5)]).clifford_conductor() # needs sage.libs.pari 30 - For hyperbolic spaces, the Clifford conductor is 1: - sage: # needs sage.libs.pari sage: H = QuadraticForm(ZZ, 2, [0, 1, 0]) sage: H.clifford_conductor() 1 sage: (H + H).clifford_conductor() 1 sage: (H + H + H).clifford_conductor() 1 sage: (H + H + H + H).clifford_conductor() 1 - >>> from sage.all import * >>> # needs sage.libs.pari >>> H = QuadraticForm(ZZ, Integer(2), [Integer(0), Integer(1), Integer(0)]) >>> H.clifford_conductor() 1 >>> (H + H).clifford_conductor() 1 >>> (H + H + H).clifford_conductor() 1 >>> (H + H + H + H).clifford_conductor() 1 
 - clifford_invariant(p)[source]¶
- Return the Clifford invariant. - This is the class in the Brauer group of the Clifford algebra for even dimension, of the even Clifford Algebra for odd dimension. - See Lam (AMS GSM 67) p. 117 for the definition, and p. 119 for the formula relating it to the Hasse invariant. - EXAMPLES: - For hyperbolic spaces, the Clifford invariant is +1: - sage: # needs sage.libs.pari sage: H = QuadraticForm(ZZ, 2, [0, 1, 0]) sage: H.clifford_invariant(2) 1 sage: (H + H).clifford_invariant(2) 1 sage: (H + H + H).clifford_invariant(2) 1 sage: (H + H + H + H).clifford_invariant(2) 1 - >>> from sage.all import * >>> # needs sage.libs.pari >>> H = QuadraticForm(ZZ, Integer(2), [Integer(0), Integer(1), Integer(0)]) >>> H.clifford_invariant(Integer(2)) 1 >>> (H + H).clifford_invariant(Integer(2)) 1 >>> (H + H + H).clifford_invariant(Integer(2)) 1 >>> (H + H + H + H).clifford_invariant(Integer(2)) 1 
 - coefficients()[source]¶
- Return the matrix of upper triangular coefficients, by reading across the rows from the main diagonal. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.coefficients() [1, 2, 3] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.coefficients() [1, 2, 3] 
 - complementary_subform_to_vector(v)[source]¶
- Find the \((n-1)\)-dimensional quadratic form orthogonal to the vector \(v\). - Note - This is usually not a direct summand! - Note - There is a minor difference in the cancellation code here (form the C++ version) since the notation - Q[i,j]indexes coefficients of the quadratic polynomial here, not the symmetric matrix. Also, it produces a better splitting now, for the full lattice (as opposed to a sublattice in the C++ code) since we now extend \(v\) to a unimodular matrix.- INPUT: - v– list of- self.dim()integers
 - OUTPUT: a - QuadraticFormover \(\ZZ\)- EXAMPLES: - sage: Q1 = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q1.complementary_subform_to_vector([1,0,0,0]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 7 0 0 ] [ * 5 0 ] [ * * 3 ] - >>> from sage.all import * >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q1.complementary_subform_to_vector([Integer(1),Integer(0),Integer(0),Integer(0)]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 7 0 0 ] [ * 5 0 ] [ * * 3 ] - sage: Q1.complementary_subform_to_vector([1,1,0,0]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 7 0 0 ] [ * 5 0 ] [ * * 12 ] - >>> from sage.all import * >>> Q1.complementary_subform_to_vector([Integer(1),Integer(1),Integer(0),Integer(0)]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 7 0 0 ] [ * 5 0 ] [ * * 12 ] - sage: Q1.complementary_subform_to_vector([1,1,1,1]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 880 -480 -160 ] [ * 624 -96 ] [ * * 240 ] - >>> from sage.all import * >>> Q1.complementary_subform_to_vector([Integer(1),Integer(1),Integer(1),Integer(1)]) Quadratic form in 3 variables over Integer Ring with coefficients: [ 880 -480 -160 ] [ * 624 -96 ] [ * * 240 ] 
 - compute_definiteness()[source]¶
- Compute whether the given quadratic form is positive-definite, negative-definite, indefinite, degenerate, or the zero form. - This caches one of the following strings in - self.__definiteness_string: “pos_def”, “neg_def”, “indef”, “zero”, “degenerate”. It is called from all routines like:- is_positive_definite(),- is_negative_definite(),- is_indefinite(), etc.- Note - A degenerate form is considered neither definite nor indefinite. - Note - The zero-dimensional form is considered both positive definite and negative definite. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,1]) sage: Q.compute_definiteness() sage: Q.is_positive_definite() True sage: Q.is_negative_definite() False sage: Q.is_indefinite() False sage: Q.is_definite() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.compute_definiteness() >>> Q.is_positive_definite() True >>> Q.is_negative_definite() False >>> Q.is_indefinite() False >>> Q.is_definite() True - sage: Q = DiagonalQuadraticForm(ZZ, []) sage: Q.compute_definiteness() sage: Q.is_positive_definite() True sage: Q.is_negative_definite() True sage: Q.is_indefinite() False sage: Q.is_definite() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, []) >>> Q.compute_definiteness() >>> Q.is_positive_definite() True >>> Q.is_negative_definite() True >>> Q.is_indefinite() False >>> Q.is_definite() True - sage: Q = DiagonalQuadraticForm(ZZ, [1,0,-1]) sage: Q.compute_definiteness() sage: Q.is_positive_definite() False sage: Q.is_negative_definite() False sage: Q.is_indefinite() False sage: Q.is_definite() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(0),-Integer(1)]) >>> Q.compute_definiteness() >>> Q.is_positive_definite() False >>> Q.is_negative_definite() False >>> Q.is_indefinite() False >>> Q.is_definite() False 
 - compute_definiteness_string_by_determinants()[source]¶
- Compute the (positive) definiteness of a quadratic form by looking at the signs of all of its upper-left subdeterminants. See also - compute_definiteness()for more documentation.- OUTPUT: string describing the definiteness - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,1]) sage: Q.compute_definiteness_string_by_determinants() 'pos_def' - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.compute_definiteness_string_by_determinants() 'pos_def' - sage: Q = DiagonalQuadraticForm(ZZ, []) sage: Q.compute_definiteness_string_by_determinants() 'zero' - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, []) >>> Q.compute_definiteness_string_by_determinants() 'zero' - sage: Q = DiagonalQuadraticForm(ZZ, [1,0,-1]) sage: Q.compute_definiteness_string_by_determinants() 'degenerate' - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(0),-Integer(1)]) >>> Q.compute_definiteness_string_by_determinants() 'degenerate' - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1]) sage: Q.compute_definiteness_string_by_determinants() 'indefinite' - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1)]) >>> Q.compute_definiteness_string_by_determinants() 'indefinite' - sage: Q = DiagonalQuadraticForm(ZZ, [-1,-1]) sage: Q.compute_definiteness_string_by_determinants() 'neg_def' - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [-Integer(1),-Integer(1)]) >>> Q.compute_definiteness_string_by_determinants() 'neg_def' 
 - content()[source]¶
- Return the GCD of the coefficients of the quadratic form. - Warning - Only works over Euclidean domains (probably just \(\ZZ\)). - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1, 1]) sage: Q.matrix().gcd() 2 sage: Q.content() 1 sage: DiagonalQuadraticForm(ZZ, [1, 1]).is_primitive() True sage: DiagonalQuadraticForm(ZZ, [2, 4]).is_primitive() False sage: DiagonalQuadraticForm(ZZ, [2, 4]).primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 2 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1)]) >>> Q.matrix().gcd() 2 >>> Q.content() 1 >>> DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1)]).is_primitive() True >>> DiagonalQuadraticForm(ZZ, [Integer(2), Integer(4)]).is_primitive() False >>> DiagonalQuadraticForm(ZZ, [Integer(2), Integer(4)]).primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 2 ] 
 - conway_cross_product_doubled_power(p)[source]¶
- Compute twice the power of \(p\) which evaluates the ‘cross product’ term in Conway’s mass formula. - INPUT: - p– a prime number > 0
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1,8)) sage: Q.conway_cross_product_doubled_power(2) 18 sage: Q.conway_cross_product_doubled_power(3) 10 sage: Q.conway_cross_product_doubled_power(5) 6 sage: Q.conway_cross_product_doubled_power(7) 6 sage: Q.conway_cross_product_doubled_power(11) 0 sage: Q.conway_cross_product_doubled_power(13) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(8))) >>> Q.conway_cross_product_doubled_power(Integer(2)) 18 >>> Q.conway_cross_product_doubled_power(Integer(3)) 10 >>> Q.conway_cross_product_doubled_power(Integer(5)) 6 >>> Q.conway_cross_product_doubled_power(Integer(7)) 6 >>> Q.conway_cross_product_doubled_power(Integer(11)) 0 >>> Q.conway_cross_product_doubled_power(Integer(13)) 0 
 - conway_diagonal_factor(p)[source]¶
- Compute the diagonal factor of Conway’s \(p\)-mass. - INPUT: - p– a prime number > 0
 - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1,6)) sage: Q.conway_diagonal_factor(3) 81/256 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(6))) >>> Q.conway_diagonal_factor(Integer(3)) 81/256 
 - conway_mass()[source]¶
- Compute the mass by using the Conway-Sloane mass formula. - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.conway_mass() # needs sage.symbolic 1/48 sage: Q = DiagonalQuadraticForm(ZZ, [7,1,1]) sage: Q.conway_mass() # needs sage.symbolic 3/16 sage: Q = QuadraticForm(ZZ, 3, [7, 2, 2, 2, 0, 2]) + DiagonalQuadraticForm(ZZ, [1]) sage: Q.conway_mass() # needs sage.symbolic 3/32 sage: Q = QuadraticForm(Matrix(ZZ, 2, [2,1,1,2])) sage: Q.conway_mass() # needs sage.symbolic 1/12 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.conway_mass() # needs sage.symbolic 1/48 >>> Q = DiagonalQuadraticForm(ZZ, [Integer(7),Integer(1),Integer(1)]) >>> Q.conway_mass() # needs sage.symbolic 3/16 >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(7), Integer(2), Integer(2), Integer(2), Integer(0), Integer(2)]) + DiagonalQuadraticForm(ZZ, [Integer(1)]) >>> Q.conway_mass() # needs sage.symbolic 3/32 >>> Q = QuadraticForm(Matrix(ZZ, Integer(2), [Integer(2),Integer(1),Integer(1),Integer(2)])) >>> Q.conway_mass() # needs sage.symbolic 1/12 
 - conway_octane_of_this_unimodular_Jordan_block_at_2()[source]¶
- Determines the ‘octane’ of this full unimodular Jordan block at the prime \(p=2\). This is an invariant defined (mod 8), ad. - This assumes that the form is given as a block diagonal form with unimodular blocks of size \(\leq 2\) and the \(1 \times 1\) blocks are all in the upper leftmost position. - OUTPUT: integer \(0 \leq x \leq 7\) - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 0 sage: Q = DiagonalQuadraticForm(ZZ, [1,5,13]) sage: Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 3 sage: Q = DiagonalQuadraticForm(ZZ, [3,7,13]) sage: Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 7 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 0 >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(5),Integer(13)]) >>> Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 3 >>> Q = DiagonalQuadraticForm(ZZ, [Integer(3),Integer(7),Integer(13)]) >>> Q.conway_octane_of_this_unimodular_Jordan_block_at_2() 7 
 - conway_p_mass(p)[source]¶
- Compute Conway’s \(p\)-mass. - INPUT: - p– a prime number > 0
 - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1, 6)) sage: Q.conway_p_mass(2) 16/3 sage: Q.conway_p_mass(3) 729/256 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1), Integer(6))) >>> Q.conway_p_mass(Integer(2)) 16/3 >>> Q.conway_p_mass(Integer(3)) 729/256 
 - conway_species_list_at_2()[source]¶
- Return an integer called the ‘species’ which determines the type of the orthogonal group over the finite field \(\GF{p}\). - This assumes that the given quadratic form is a unimodular Jordan block at an odd prime \(p\). When the dimension is odd then this number is always positive, otherwise it may be positive or negative. - Note - The species of a zero dimensional form is always 0+, so we interpret the return value of zero as positive here! =) - OUTPUT: list of integers - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1,10)) sage: Q.conway_species_list_at_2() [1, 5, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(10))) >>> Q.conway_species_list_at_2() [1, 5, 1, 1, 1, 1] - sage: Q = DiagonalQuadraticForm(ZZ, range(1,8)) sage: Q.conway_species_list_at_2() [1, 3, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(8))) >>> Q.conway_species_list_at_2() [1, 3, 1, 1, 1] 
 - conway_species_list_at_odd_prime(p)[source]¶
- Return an integer called the ‘species’ which determines the type of the orthogonal group over the finite field \(\GF{p}\). - This assumes that the given quadratic form is a unimodular Jordan block at an odd prime \(p\). When the dimension is odd then this number is always positive, otherwise it may be positive or negative (or zero, but that is considered positive by convention). - Note - The species of a zero dimensional form is always 0+, so we interpret the return value of zero as positive here! =) - INPUT: - p– a positive prime number
 - OUTPUT: list of integers - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1,10)) sage: Q.conway_species_list_at_odd_prime(3) [6, 2, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(10))) >>> Q.conway_species_list_at_odd_prime(Integer(3)) [6, 2, 1] - sage: Q = DiagonalQuadraticForm(ZZ, range(1,8)) sage: Q.conway_species_list_at_odd_prime(3) [5, 2] sage: Q.conway_species_list_at_odd_prime(5) [-6, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(8))) >>> Q.conway_species_list_at_odd_prime(Integer(3)) [5, 2] >>> Q.conway_species_list_at_odd_prime(Integer(5)) [-6, 1] 
 - conway_standard_mass()[source]¶
- Return the infinite product of the standard mass factors. - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [2, -2, 0, 3, -5, 4]) sage: Q.conway_standard_mass() # needs sage.symbolic 1/6 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(2), Integer(0), Integer(3), -Integer(5), Integer(4)]) >>> Q.conway_standard_mass() # needs sage.symbolic 1/6 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.conway_standard_mass() # needs sage.symbolic 1/6 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.conway_standard_mass() # needs sage.symbolic 1/6 
 - conway_standard_p_mass(p)[source]¶
- Compute the standard (generic) Conway-Sloane \(p\)-mass. - INPUT: - p– a prime number > 0
 - OUTPUT: a rational number > 0 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.conway_standard_p_mass(2) 2/3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.conway_standard_p_mass(Integer(2)) 2/3 
 - conway_type_factor()[source]¶
- This is a special factor only present in the mass formula when \(p=2\). - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, range(1,8)) sage: Q.conway_type_factor() 4 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, range(Integer(1),Integer(8))) >>> Q.conway_type_factor() 4 
 - count_congruence_solutions(p, k, m, zvec, nzvec)[source]¶
- Count all solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers in- range(self.dim()), or- None
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions(3, 1, 0, None, None) 15 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions(Integer(3), Integer(1), Integer(0), None, None) 15 
 - count_congruence_solutions__bad_type(p, k, m, zvec, nzvec)[source]¶
- Count the bad-type solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers up to dim(\(Q\))
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions__bad_type(3, 1, 0, None, None) 2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions__bad_type(Integer(3), Integer(1), Integer(0), None, None) 2 
 - count_congruence_solutions__bad_type_I(p, k, m, zvec, nzvec)[source]¶
- Count the bad-typeI solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers up to dim(\(Q\))
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions__bad_type_I(3, 1, 0, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions__bad_type_I(Integer(3), Integer(1), Integer(0), None, None) 0 
 - count_congruence_solutions__bad_type_II(p, k, m, zvec, nzvec)[source]¶
- Count the bad-typeII solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers up to dim(\(Q\))
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions__bad_type_II(3, 1, 0, None, None) 2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions__bad_type_II(Integer(3), Integer(1), Integer(0), None, None) 2 
 - count_congruence_solutions__good_type(p, k, m, zvec, nzvec)[source]¶
- Count the good-type solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers up to dim(\(Q\))
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions__good_type(3, 1, 0, None, None) 12 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions__good_type(Integer(3), Integer(1), Integer(0), None, None) 12 
 - count_congruence_solutions__zero_type(p, k, m, zvec, nzvec)[source]¶
- Count the zero-type solutions of \(Q(x) = m\) (mod \(p^k\)) satisfying the additional congruence conditions described in - QuadraticForm.count_congruence_solutions_as_vector().- INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers up to dim(\(Q\))
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions__zero_type(3, 1, 0, None, None) 1 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions__zero_type(Integer(3), Integer(1), Integer(0), None, None) 1 
 - count_congruence_solutions_as_vector(p, k, m, zvec, nzvec)[source]¶
- Return the number of integer solution vectors \(x\) satisfying the congruence \(Q(x) = m\) (mod \(p^k\)) of each solution type (i.e. All, Good, Zero, Bad, BadI, BadII) which satisfy the additional congruence conditions of having certain coefficients = 0 (mod \(p\)) and certain collections of coefficients not congruent to the zero vector (mod \(p\)). - A solution vector \(x\) satisfies the additional congruence conditions specified by - zvecand- nzvec(and therefore is counted) iff both of the following conditions hold:- \(x_i = 0\) (mod \(p\)) for all \(i\) in - zvec
- \(x_i \neq 0\) (mod \(p\)) for all \(i\) in - nzvec
 - REFERENCES: - See Hanke’s (????) paper “Local Densities and explicit bounds…”, p??? for the definitions of the solution types and congruence conditions. - INPUT: - p– prime number > 0
- k– integer > 0
- m– integer (depending only on mod \(p^k\))
- zvec,- nzvec– lists of integers in- range(self.dim()), or- None
 - OUTPUT: - a list of six integers \(\geq 0\) representing the solution types: [All, Good, Zero, Bad, BadI, BadII] - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.count_congruence_solutions_as_vector(3, 1, 1, [], []) [0, 0, 0, 0, 0, 0] sage: Q.count_congruence_solutions_as_vector(3, 1, 1, None, []) [0, 0, 0, 0, 0, 0] sage: Q.count_congruence_solutions_as_vector(3, 1, 1, [], None) [6, 6, 0, 0, 0, 0] sage: Q.count_congruence_solutions_as_vector(3, 1, 1, None, None) [6, 6, 0, 0, 0, 0] sage: Q.count_congruence_solutions_as_vector(3, 1, 2, None, None) [6, 6, 0, 0, 0, 0] sage: Q.count_congruence_solutions_as_vector(3, 1, 0, None, None) [15, 12, 1, 2, 0, 2] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(1), [], []) [0, 0, 0, 0, 0, 0] >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(1), None, []) [0, 0, 0, 0, 0, 0] >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(1), [], None) [6, 6, 0, 0, 0, 0] >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(1), None, None) [6, 6, 0, 0, 0, 0] >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(2), None, None) [6, 6, 0, 0, 0, 0] >>> Q.count_congruence_solutions_as_vector(Integer(3), Integer(1), Integer(0), None, None) [15, 12, 1, 2, 0, 2] 
 - count_modp_solutions__by_Gauss_sum(p, m)[source]¶
- Return the number of solutions of \(Q(x) = m\) (mod \(p\)) of a non-degenerate quadratic form over the finite field \(\ZZ/p\ZZ\), where \(p\) is a prime number > 2. - Note - We adopt the useful convention that a zero-dimensional quadratic form has exactly one solution always (i.e. the empty vector). - These are defined in Table 1 on p363 of Hanke’s “Local Densities…” paper. - INPUT: - p– a prime number > 2
- m– integer
 - OUTPUT: integer \(\geq 0\) - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: [Q.count_modp_solutions__by_Gauss_sum(3, m) for m in range(3)] [9, 6, 12] sage: Q = DiagonalQuadraticForm(ZZ, [1,1,2]) sage: [Q.count_modp_solutions__by_Gauss_sum(3, m) for m in range(3)] [9, 12, 6] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> [Q.count_modp_solutions__by_Gauss_sum(Integer(3), m) for m in range(Integer(3))] [9, 6, 12] >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(2)]) >>> [Q.count_modp_solutions__by_Gauss_sum(Integer(3), m) for m in range(Integer(3))] [9, 12, 6] 
 - delta()[source]¶
- Return the omega of the adjoint form. - This is the same as the omega of the reciprocal form. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,37]) sage: Q.delta() 148 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(37)]) >>> Q.delta() 148 
 - det()[source]¶
- Return the determinant of the Gram matrix of \(2\cdot Q\), or equivalently the determinant of the Hessian matrix of \(Q\). - Note - This is always defined over the same ring as the quadratic form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.det() 8 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.det() 8 
 - dim()[source]¶
- Return the number of variables of the quadratic form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.dim() 2 sage: parent(Q.dim()) Integer Ring sage: Q = QuadraticForm(Q.matrix()) sage: Q.dim() 2 sage: parent(Q.dim()) Integer Ring - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.dim() 2 >>> parent(Q.dim()) Integer Ring >>> Q = QuadraticForm(Q.matrix()) >>> Q.dim() 2 >>> parent(Q.dim()) Integer Ring 
 - disc()[source]¶
- Return the discriminant of the quadratic form, defined as - \((-1)^n {\rm det}(B)\) for even dimension \(2n\) 
- \({\rm det}(B)/2\) for odd dimension 
 - where \(2Q(x) = x^t B x\). - This agrees with the usual discriminant for binary and ternary quadratic forms. - EXAMPLES: - sage: DiagonalQuadraticForm(ZZ, [1]).disc() 1 sage: DiagonalQuadraticForm(ZZ, [1,1]).disc() -4 sage: DiagonalQuadraticForm(ZZ, [1,1,1]).disc() 4 sage: DiagonalQuadraticForm(ZZ, [1,1,1,1]).disc() 16 - >>> from sage.all import * >>> DiagonalQuadraticForm(ZZ, [Integer(1)]).disc() 1 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]).disc() -4 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]).disc() 4 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]).disc() 16 
 - discrec()[source]¶
- Return the discriminant of the reciprocal form. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,37]) sage: Q.disc() 148 sage: Q.discrec() 5476 sage: [4 * 37, 4 * 37^2] [148, 5476] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(37)]) >>> Q.disc() 148 >>> Q.discrec() 5476 >>> [Integer(4) * Integer(37), Integer(4) * Integer(37)**Integer(2)] [148, 5476] 
 - divide_variable(c, i, in_place=False)[source]¶
- Replace the variables \(x_i\) by \((x_i)/c\) in the quadratic form (replacing the original form if the - in_placeflag is True).- Here \(c\) must be an element of the base ring defining the quadratic form, and the division must be defined in the base ring. - INPUT: - c– an element of- self.base_ring()
- i– integer \(\geq 0\)
 - OUTPUT: - a - QuadraticForm(by default, otherwise none)- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,9,5,7]) sage: Q.divide_variable(3, 1) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 0 0 ] [ * * 5 0 ] [ * * * 7 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(9),Integer(5),Integer(7)]) >>> Q.divide_variable(Integer(3), Integer(1)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 0 0 ] [ * * 5 0 ] [ * * * 7 ] 
 - elementary_substitution(c, i, j, in_place=False)[source]¶
- Perform the substitution \(x_i \longmapsto x_i + c\cdot x_j\) (replacing the original form if the - in_placeflag is True).- INPUT: - c– an element of- self.base_ring()
- i,- j– integers \(\geq 0\)
 - OUTPUT: - a - QuadraticForm(by default, otherwise none)- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(1,11)); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] sage: Q.elementary_substitution(c=1, i=0, j=3) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 6 ] [ * 5 6 9 ] [ * * 8 12 ] [ * * * 15 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(1),Integer(11))); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] >>> Q.elementary_substitution(c=Integer(1), i=Integer(0), j=Integer(3)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 6 ] [ * 5 6 9 ] [ * * 8 12 ] [ * * * 15 ] - sage: R = QuadraticForm(ZZ, 4, range(1,11)); R Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] - >>> from sage.all import * >>> R = QuadraticForm(ZZ, Integer(4), range(Integer(1),Integer(11))); R Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] - sage: M = Matrix(ZZ, 4, 4, [1,0,0,1, 0,1,0,0, 0,0,1,0, 0,0,0,1]); M [1 0 0 1] [0 1 0 0] [0 0 1 0] [0 0 0 1] sage: R(M) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 6 ] [ * 5 6 9 ] [ * * 8 12 ] [ * * * 15 ] - >>> from sage.all import * >>> M = Matrix(ZZ, Integer(4), Integer(4), [Integer(1),Integer(0),Integer(0),Integer(1), Integer(0),Integer(1),Integer(0),Integer(0), Integer(0),Integer(0),Integer(1),Integer(0), Integer(0),Integer(0),Integer(0),Integer(1)]); M [1 0 0 1] [0 1 0 0] [0 0 1 0] [0 0 0 1] >>> R(M) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 6 ] [ * 5 6 9 ] [ * * 8 12 ] [ * * * 15 ] 
 - extract_variables(QF, var_indices)[source]¶
- Extract the variables (in order) whose indices are listed in - var_indices, to give a new quadratic form.- INPUT: - var_indices– list of integers \(\geq 0\)
 - OUTPUT: a - QuadraticForm- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(10)); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] sage: Q.extract_variables([1,3]) Quadratic form in 2 variables over Integer Ring with coefficients: [ 4 6 ] [ * 9 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] >>> Q.extract_variables([Integer(1),Integer(3)]) Quadratic form in 2 variables over Integer Ring with coefficients: [ 4 6 ] [ * 9 ] 
 - find_entry_with_minimal_scale_at_prime(p)[source]¶
- Find the entry of the quadratic form with minimal scale at the prime \(p\), preferring diagonal entries in case of a tie. - (I.e. If we write the quadratic form as a symmetric matrix \(M\), then this entry - M[i,j]has the minimal valuation at the prime \(p\).)- Note - This answer is independent of the kind of matrix (Gram or Hessian) associated to the form. - INPUT: - p– a prime number > 0
 - OUTPUT: a pair of integers \(\geq 0\) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [6, 2, 20]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 6 2 ] [ * 20 ] sage: Q.find_entry_with_minimal_scale_at_prime(2) (0, 1) sage: Q.find_entry_with_minimal_scale_at_prime(3) (1, 1) sage: Q.find_entry_with_minimal_scale_at_prime(5) (0, 0) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(6), Integer(2), Integer(20)]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 6 2 ] [ * 20 ] >>> Q.find_entry_with_minimal_scale_at_prime(Integer(2)) (0, 1) >>> Q.find_entry_with_minimal_scale_at_prime(Integer(3)) (1, 1) >>> Q.find_entry_with_minimal_scale_at_prime(Integer(5)) (0, 0) 
 - find_p_neighbor_from_vec(p, y, return_matrix=False)[source]¶
- Return the \(p\)-neighbor of - selfdefined by- y.- Let \((L,q)\) be a lattice with \(b(L,L) \subseteq \ZZ\) which is maximal at \(p\). Let \(y \in L\) with \(b(y,y) \in p^2\ZZ\) then the \(p\)-neighbor of \(L\) at \(y\) is given by \(\ZZ y/p + L_y\) where \(L_y = \{x \in L | b(x,y) \in p \ZZ \}\) and \(b(x,y) = q(x+y)-q(x)-q(y)\) is the bilinear form associated to \(q\). - INPUT: - p– a prime number
- y– a vector with \(q(y) \in p \ZZ\)
- odd– boolean (default:- False); if \(p=2\), return also odd neighbors
- return_matrix– boolean (default:- False); return the transformation matrix instead of the quadratic form
 - EXAMPLES: - sage: # needs sage.libs.pari sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: v = vector([0,2,1,1]) sage: X = Q.find_p_neighbor_from_vec(3, v); X Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 4 4 ] [ * * 5 12 ] [ * * * 9 ] sage: B = Q.find_p_neighbor_from_vec(3, v, return_matrix=True) sage: Q(B) == X True - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> v = vector([Integer(0),Integer(2),Integer(1),Integer(1)]) >>> X = Q.find_p_neighbor_from_vec(Integer(3), v); X Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 4 4 ] [ * * 5 12 ] [ * * * 9 ] >>> B = Q.find_p_neighbor_from_vec(Integer(3), v, return_matrix=True) >>> Q(B) == X True - Since the base ring and the domain are not yet separate, for rational, half integral forms we just pretend the base ring is \(\ZZ\): - sage: # needs sage.libs.pari sage: Q = QuadraticForm(QQ, matrix.diagonal([1,1,1,1])) sage: v = vector([1,1,1,1]) sage: Q.find_p_neighbor_from_vec(2, v) Quadratic form in 4 variables over Rational Field with coefficients: [ 1/2 1 1 1 ] [ * 1 1 2 ] [ * * 1 2 ] [ * * * 2 ] - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = QuadraticForm(QQ, matrix.diagonal([Integer(1),Integer(1),Integer(1),Integer(1)])) >>> v = vector([Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.find_p_neighbor_from_vec(Integer(2), v) Quadratic form in 4 variables over Rational Field with coefficients: [ 1/2 1 1 1 ] [ * 1 1 2 ] [ * * 1 2 ] [ * * * 2 ] 
 - find_primitive_p_divisible_vector__next(p, v=None)[source]¶
- Find the next \(p\)-primitive vector (up to scaling) in \(L/pL\) whose value is \(p\)-divisible, where the last vector returned was \(v\). For an initial call, no \(v\) needs to be passed. - Return vectors whose last nonzero entry is normalized to 0 or 1 (so no lines are counted repeatedly). The ordering is by increasing the first non-normalized entry. If we have tested all (lines of) vectors, then return None. - OUTPUT: vector or None - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [10,1,4]) sage: v = Q.find_primitive_p_divisible_vector__next(5); v (1, 1) sage: v = Q.find_primitive_p_divisible_vector__next(5, v); v (1, 0) sage: v = Q.find_primitive_p_divisible_vector__next(5, v); v sage: v = Q.find_primitive_p_divisible_vector__next(2) ; v (0, 1) sage: v = Q.find_primitive_p_divisible_vector__next(2, v) ; v (1, 0) sage: Q = QuadraticForm(QQ, matrix.diagonal([1,1,1,1])) sage: v = Q.find_primitive_p_divisible_vector__next(2) sage: Q(v) 2 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(10),Integer(1),Integer(4)]) >>> v = Q.find_primitive_p_divisible_vector__next(Integer(5)); v (1, 1) >>> v = Q.find_primitive_p_divisible_vector__next(Integer(5), v); v (1, 0) >>> v = Q.find_primitive_p_divisible_vector__next(Integer(5), v); v >>> v = Q.find_primitive_p_divisible_vector__next(Integer(2)) ; v (0, 1) >>> v = Q.find_primitive_p_divisible_vector__next(Integer(2), v) ; v (1, 0) >>> Q = QuadraticForm(QQ, matrix.diagonal([Integer(1),Integer(1),Integer(1),Integer(1)])) >>> v = Q.find_primitive_p_divisible_vector__next(Integer(2)) >>> Q(v) 2 
 - find_primitive_p_divisible_vector__random(p)[source]¶
- Find a random \(p\)-primitive vector in \(L/pL\) whose value is \(p\)-divisible. - Note - Since there are about \(p^{(n-2)}\) of these lines, we have a \(1/p\) chance of randomly finding an appropriate vector. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [10,1,4]) sage: v = Q.find_primitive_p_divisible_vector__random(5) sage: tuple(v) in ((1, 0), (1, 1), (2, 0), (2, 2), (3, 0), (3, 3), (4, 0), (4, 4)) True sage: 5.divides(Q(v)) True sage: Q = QuadraticForm(QQ, matrix.diagonal([1,1,1,1])) sage: v = Q.find_primitive_p_divisible_vector__random(2) sage: Q(v) 2 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(10),Integer(1),Integer(4)]) >>> v = Q.find_primitive_p_divisible_vector__random(Integer(5)) >>> tuple(v) in ((Integer(1), Integer(0)), (Integer(1), Integer(1)), (Integer(2), Integer(0)), (Integer(2), Integer(2)), (Integer(3), Integer(0)), (Integer(3), Integer(3)), (Integer(4), Integer(0)), (Integer(4), Integer(4))) True >>> Integer(5).divides(Q(v)) True >>> Q = QuadraticForm(QQ, matrix.diagonal([Integer(1),Integer(1),Integer(1),Integer(1)])) >>> v = Q.find_primitive_p_divisible_vector__random(Integer(2)) >>> Q(v) 2 
 - static from_polynomial(poly)[source]¶
- Construct a - QuadraticFormfrom a multivariate polynomial. Inverse of- polynomial().- EXAMPLES: - sage: R.<x,y,z> = ZZ[] sage: f = 5*x^2 - x*z - 3*y*z - 2*y^2 + 9*z^2 sage: Q = QuadraticForm.from_polynomial(f); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 5 0 -1 ] [ * -2 -3 ] [ * * 9 ] sage: Q.polynomial() 5*x0^2 - 2*x1^2 - x0*x2 - 3*x1*x2 + 9*x2^2 sage: Q.polynomial()(R.gens()) == f True - >>> from sage.all import * >>> R = ZZ['x, y, z']; (x, y, z,) = R._first_ngens(3) >>> f = Integer(5)*x**Integer(2) - x*z - Integer(3)*y*z - Integer(2)*y**Integer(2) + Integer(9)*z**Integer(2) >>> Q = QuadraticForm.from_polynomial(f); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 5 0 -1 ] [ * -2 -3 ] [ * * 9 ] >>> Q.polynomial() 5*x0^2 - 2*x1^2 - x0*x2 - 3*x1*x2 + 9*x2^2 >>> Q.polynomial()(R.gens()) == f True - The method fails if the given polynomial is not a quadratic form: - sage: QuadraticForm.from_polynomial(x^3 + x*z + 5*y^2) Traceback (most recent call last): ... ValueError: polynomial has monomials of degree != 2 - >>> from sage.all import * >>> QuadraticForm.from_polynomial(x**Integer(3) + x*z + Integer(5)*y**Integer(2)) Traceback (most recent call last): ... ValueError: polynomial has monomials of degree != 2 
 - gcd()[source]¶
- Return the greatest common divisor of the coefficients of the quadratic form (as a polynomial). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(1, 21, 2)) sage: Q.gcd() 1 sage: Q = QuadraticForm(ZZ, 4, range(0, 20, 2)) sage: Q.gcd() 2 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(1), Integer(21), Integer(2))) >>> Q.gcd() 1 >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(0), Integer(20), Integer(2))) >>> Q.gcd() 2 
 - static genera(sig_pair, determinant, max_scale=None, even=False)[source]¶
- Return a list of all global genera with the given conditions. - Here a genus is called global if it is non-empty. - INPUT: - sig_pair– a pair of nonnegative integers giving the signature
- determinant– integer; the sign is ignored
- max_scale– (default:- None) an integer; the maximum scale of a jordan block
- even– boolean (default:- False)
 - OUTPUT: - A list of all (non-empty) global genera with the given conditions. - EXAMPLES: - sage: QuadraticForm.genera((4,0), 125, even=True) [Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^1 5^3, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^-2 5^1 25^-1, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^2 5^1 25^1, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^3 125^1] - >>> from sage.all import * >>> QuadraticForm.genera((Integer(4),Integer(0)), Integer(125), even=True) [Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^1 5^3, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^-2 5^1 25^-1, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^2 5^1 25^1, Genus of None Signature: (4, 0) Genus symbol at 2: 1^-4 Genus symbol at 5: 1^3 125^1] 
 - global_genus_symbol()[source]¶
- Return the genus of two times a quadratic form over \(\ZZ\). - These are defined by a collection of local genus symbols (a la Chapter 15 of Conway-Sloane [CS1999]), and a signature. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3,4]) sage: Q.global_genus_symbol() Genus of [2 0 0 0] [0 4 0 0] [0 0 6 0] [0 0 0 8] Signature: (4, 0) Genus symbol at 2: [2^-2 4^1 8^1]_6 Genus symbol at 3: 1^3 3^-1 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3),Integer(4)]) >>> Q.global_genus_symbol() Genus of [2 0 0 0] [0 4 0 0] [0 0 6 0] [0 0 0 8] Signature: (4, 0) Genus symbol at 2: [2^-2 4^1 8^1]_6 Genus symbol at 3: 1^3 3^-1 - sage: Q = QuadraticForm(ZZ, 4, range(10)) sage: Q.global_genus_symbol() Genus of [ 0 1 2 3] [ 1 8 5 6] [ 2 5 14 8] [ 3 6 8 18] Signature: (3, 1) Genus symbol at 2: 1^-4 Genus symbol at 563: 1^3 563^-1 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))) >>> Q.global_genus_symbol() Genus of [ 0 1 2 3] [ 1 8 5 6] [ 2 5 14 8] [ 3 6 8 18] Signature: (3, 1) Genus symbol at 2: 1^-4 Genus symbol at 563: 1^3 563^-1 
 - has_equivalent_Jordan_decomposition_at_prime(other, p)[source]¶
- Determine if the given quadratic form has a Jordan decomposition equivalent to that of - self.- INPUT: - other– a- QuadraticForm
 - OUTPUT: boolean - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 1, 0, 3]) sage: Q2 = QuadraticForm(ZZ, 3, [1, 0, 0, 2, -2, 6]) sage: Q3 = QuadraticForm(ZZ, 3, [1, 0, 0, 1, 0, 11]) sage: [Q1.level(), Q2.level(), Q3.level()] [44, 44, 44] sage: # needs sage.libs.pari sage: Q1.has_equivalent_Jordan_decomposition_at_prime(Q2, 2) False sage: Q1.has_equivalent_Jordan_decomposition_at_prime(Q2, 11) False sage: Q1.has_equivalent_Jordan_decomposition_at_prime(Q3, 2) False sage: Q1.has_equivalent_Jordan_decomposition_at_prime(Q3, 11) True sage: Q2.has_equivalent_Jordan_decomposition_at_prime(Q3, 2) True sage: Q2.has_equivalent_Jordan_decomposition_at_prime(Q3, 11) False - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(1), Integer(0), Integer(3)]) >>> Q2 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), Integer(0), Integer(2), -Integer(2), Integer(6)]) >>> Q3 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), Integer(0), Integer(1), Integer(0), Integer(11)]) >>> [Q1.level(), Q2.level(), Q3.level()] [44, 44, 44] >>> # needs sage.libs.pari >>> Q1.has_equivalent_Jordan_decomposition_at_prime(Q2, Integer(2)) False >>> Q1.has_equivalent_Jordan_decomposition_at_prime(Q2, Integer(11)) False >>> Q1.has_equivalent_Jordan_decomposition_at_prime(Q3, Integer(2)) False >>> Q1.has_equivalent_Jordan_decomposition_at_prime(Q3, Integer(11)) True >>> Q2.has_equivalent_Jordan_decomposition_at_prime(Q3, Integer(2)) True >>> Q2.has_equivalent_Jordan_decomposition_at_prime(Q3, Integer(11)) False 
 - has_integral_Gram_matrix()[source]¶
- Return whether the quadratic form has an integral Gram matrix (with respect to its base ring). - A warning is issued if the form is defined over a field, since in that case the return is trivially true. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [7,8,9]) sage: Q.has_integral_Gram_matrix() True - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(7),Integer(8),Integer(9)]) >>> Q.has_integral_Gram_matrix() True - sage: Q = QuadraticForm(ZZ, 2, [4,5,6]) sage: Q.has_integral_Gram_matrix() False - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(4),Integer(5),Integer(6)]) >>> Q.has_integral_Gram_matrix() False 
 - hasse_conductor()[source]¶
- Return the Hasse conductor. - This is the product of all primes where the Hasse invariant equals \(-1\). - EXAMPLES: - sage: # needs sage.libs.pari sage: Q = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q.hasse_invariant(2) -1 sage: Q.hasse_invariant(37) -1 sage: Q.hasse_conductor() 74 sage: DiagonalQuadraticForm(ZZ, [1, 1, 1]).hasse_conductor() # needs sage.libs.pari 1 sage: QuadraticForm(ZZ, 3, [2, -2, 0, 2, 0, 5]).hasse_conductor() # needs sage.libs.pari 10 - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q.hasse_invariant(Integer(2)) -1 >>> Q.hasse_invariant(Integer(37)) -1 >>> Q.hasse_conductor() 74 >>> DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1), Integer(1)]).hasse_conductor() # needs sage.libs.pari 1 >>> QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(2), Integer(0), Integer(2), Integer(0), Integer(5)]).hasse_conductor() # needs sage.libs.pari 10 
 - hasse_invariant(p)[source]¶
- Compute the Hasse invariant at a prime \(p\) or at infinity, as given on p55 of Cassels’s book. If \(Q\) is diagonal with coefficients \(a_i\), then the (Cassels) Hasse invariant is given by \[c_p = \prod_{i < j} (a_i, a_j)_p\]- where \((a,b)_p\) is the Hilbert symbol at \(p\). The underlying quadratic form must be non-degenerate over \(\QQ_p\) for this to make sense. - Warning - This is different from the O’Meara Hasse invariant, which allows \(i \leq j\) in the product. That is given by the method - hasse_invariant__OMeara().- Note - We should really rename this - hasse_invariant__Cassels, and set- hasse_invariant()as a front-end to it.- INPUT: - p– a prime number > 0 or \(-1\) for the infinite place
 - OUTPUT: \(1\) or \(-1\) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * 2 ] sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * 2 ] >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1]) sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1)]) >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1,5]) sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1),Integer(5)]) >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 - 23) # needs sage.rings.number_field sage: Q = DiagonalQuadraticForm(K, [-a, a + 2]) # needs sage.rings.number_field sage: [Q.hasse_invariant(p) for p in K.primes_above(19)] # needs sage.rings.number_field [-1, 1] - >>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(23), names=('a',)); (a,) = K._first_ngens(1)# needs sage.rings.number_field >>> Q = DiagonalQuadraticForm(K, [-a, a + Integer(2)]) # needs sage.rings.number_field >>> [Q.hasse_invariant(p) for p in K.primes_above(Integer(19))] # needs sage.rings.number_field [-1, 1] 
 - hasse_invariant__OMeara(p)[source]¶
- Compute the O’Meara Hasse invariant at a prime \(p\). - This is defined on p167 of O’Meara’s book. If \(Q\) is diagonal with coefficients \(a_i\), then the (Cassels) Hasse invariant is given by \[c_p = \prod_{i \leq j} (a_i, a_j)_p\]- where \((a,b)_p\) is the Hilbert symbol at \(p\). - Warning - This is different from the (Cassels) Hasse invariant, which only allows \(i < j\) in the product. That is given by the method hasse_invariant(p). - INPUT: - p– a prime number > 0 or \(-1\) for the infinite place
 - OUTPUT: \(1\) or \(-1\) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * 2 ] sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * 2 ] >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1]) sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1)]) >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - sage: Q = DiagonalQuadraticForm(ZZ,[1,-1,-1]) sage: [Q.hasse_invariant(p) for p in prime_range(20)] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] sage: [Q.hasse_invariant__OMeara(p) for p in prime_range(20)] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ,[Integer(1),-Integer(1),-Integer(1)]) >>> [Q.hasse_invariant(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] >>> [Q.hasse_invariant__OMeara(p) for p in prime_range(Integer(20))] # needs sage.libs.pari [-1, 1, 1, 1, 1, 1, 1, 1] - sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 - 23) # needs sage.rings.number_field sage: Q = DiagonalQuadraticForm(K, [-a, a + 2]) # needs sage.rings.number_field sage: [Q.hasse_invariant__OMeara(p) for p in K.primes_above(19)] # needs sage.rings.number_field [1, 1] - >>> from sage.all import * >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(23), names=('a',)); (a,) = K._first_ngens(1)# needs sage.rings.number_field >>> Q = DiagonalQuadraticForm(K, [-a, a + Integer(2)]) # needs sage.rings.number_field >>> [Q.hasse_invariant__OMeara(p) for p in K.primes_above(Integer(19))] # needs sage.rings.number_field [1, 1] 
 - is_adjoint()[source]¶
- Determine if the given form is the adjoint of another form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q.is_adjoint() # needs sage.symbolic False sage: Q.adjoint().is_adjoint() True - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q.is_adjoint() # needs sage.symbolic False >>> Q.adjoint().is_adjoint() True 
 - is_anisotropic(p)[source]¶
- Check if the quadratic form is anisotropic over the \(p\)-adic numbers \(\QQ_p\) or \(\RR\). - INPUT: - p– a prime number > 0 or \(-1\) for the infinite place
 - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1]) sage: Q.is_anisotropic(2) # needs sage.libs.pari True sage: Q.is_anisotropic(3) # needs sage.libs.pari True sage: Q.is_anisotropic(5) # needs sage.libs.pari False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]) >>> Q.is_anisotropic(Integer(2)) # needs sage.libs.pari True >>> Q.is_anisotropic(Integer(3)) # needs sage.libs.pari True >>> Q.is_anisotropic(Integer(5)) # needs sage.libs.pari False - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1]) sage: Q.is_anisotropic(2) # needs sage.libs.pari False sage: Q.is_anisotropic(3) # needs sage.libs.pari False sage: Q.is_anisotropic(5) # needs sage.libs.pari False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1)]) >>> Q.is_anisotropic(Integer(2)) # needs sage.libs.pari False >>> Q.is_anisotropic(Integer(3)) # needs sage.libs.pari False >>> Q.is_anisotropic(Integer(5)) # needs sage.libs.pari False - sage: [DiagonalQuadraticForm(ZZ, # needs sage.libs.pari ....: [1, -least_quadratic_nonresidue(p)]).is_anisotropic(p) ....: for p in prime_range(3, 30)] [True, True, True, True, True, True, True, True, True] - >>> from sage.all import * >>> [DiagonalQuadraticForm(ZZ, # needs sage.libs.pari ... [Integer(1), -least_quadratic_nonresidue(p)]).is_anisotropic(p) ... for p in prime_range(Integer(3), Integer(30))] [True, True, True, True, True, True, True, True, True] - sage: [DiagonalQuadraticForm(ZZ, [1, -least_quadratic_nonresidue(p), # needs sage.libs.pari ....: p, -p*least_quadratic_nonresidue(p)]).is_anisotropic(p) ....: for p in prime_range(3, 30)] [True, True, True, True, True, True, True, True, True] - >>> from sage.all import * >>> [DiagonalQuadraticForm(ZZ, [Integer(1), -least_quadratic_nonresidue(p), # needs sage.libs.pari ... p, -p*least_quadratic_nonresidue(p)]).is_anisotropic(p) ... for p in prime_range(Integer(3), Integer(30))] [True, True, True, True, True, True, True, True, True] 
 - is_definite()[source]¶
- Determines if the given quadratic form is (positive or negative) definite. - Note - A degenerate form is considered neither definite nor indefinite. - Note - The zero-dimensional form is considered indefinite. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [-1,-3,-5]) sage: Q.is_definite() True sage: Q = DiagonalQuadraticForm(ZZ, [1,-3,5]) sage: Q.is_definite() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [-Integer(1),-Integer(3),-Integer(5)]) >>> Q.is_definite() True >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(3),Integer(5)]) >>> Q.is_definite() False 
 - is_even(allow_rescaling_flag=True)[source]¶
- Return true iff after rescaling by some appropriate factor, the form represents no odd integers. For more details, see - parity().- Requires that \(Q\) is defined over \(\ZZ\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1, 0, 1]) sage: Q.is_even() False sage: Q = QuadraticForm(ZZ, 2, [1, 1, 1]) sage: Q.is_even() True - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1), Integer(0), Integer(1)]) >>> Q.is_even() False >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1), Integer(1), Integer(1)]) >>> Q.is_even() True 
 - is_globally_equivalent_to(other, return_matrix=False)[source]¶
- Determine if the current quadratic form is equivalent to the given form over \(\ZZ\). - If - return_matrixis True, then we return the transformation matrix \(M\) so that- self(M) == other.- INPUT: - self,- other– positive definite integral quadratic forms
- return_matrix– boolean (default:- False); return the transformation matrix instead of a boolean
 - OUTPUT: - if - return_matrixis- False: a boolean
- if - return_matrixis- True: either- Falseor the transformation matrix
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: M = Matrix(ZZ, 4, 4, [1,2,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1]) sage: Q1 = Q(M) sage: Q.is_globally_equivalent_to(Q1) # needs sage.libs.pari True sage: MM = Q.is_globally_equivalent_to(Q1, return_matrix=True) # needs sage.libs.pari sage: Q(MM) == Q1 # needs sage.libs.pari True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> M = Matrix(ZZ, Integer(4), Integer(4), [Integer(1),Integer(2),Integer(0),Integer(0), Integer(0),Integer(1),Integer(0),Integer(0), Integer(0),Integer(0),Integer(1),Integer(0), Integer(0),Integer(0),Integer(0),Integer(1)]) >>> Q1 = Q(M) >>> Q.is_globally_equivalent_to(Q1) # needs sage.libs.pari True >>> MM = Q.is_globally_equivalent_to(Q1, return_matrix=True) # needs sage.libs.pari >>> Q(MM) == Q1 # needs sage.libs.pari True - sage: # needs sage.libs.pari sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q2 = QuadraticForm(ZZ, 3, [2, 1, 2, 2, 1, 3]) sage: Q3 = QuadraticForm(ZZ, 3, [8, 6, 5, 3, 4, 2]) sage: Q1.is_globally_equivalent_to(Q2) False sage: Q1.is_globally_equivalent_to(Q2, return_matrix=True) False sage: Q1.is_globally_equivalent_to(Q3) True sage: M = Q1.is_globally_equivalent_to(Q3, True); M [-1 -1 0] [ 1 1 1] [-1 0 0] sage: Q1(M) == Q3 True - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q2 = QuadraticForm(ZZ, Integer(3), [Integer(2), Integer(1), Integer(2), Integer(2), Integer(1), Integer(3)]) >>> Q3 = QuadraticForm(ZZ, Integer(3), [Integer(8), Integer(6), Integer(5), Integer(3), Integer(4), Integer(2)]) >>> Q1.is_globally_equivalent_to(Q2) False >>> Q1.is_globally_equivalent_to(Q2, return_matrix=True) False >>> Q1.is_globally_equivalent_to(Q3) True >>> M = Q1.is_globally_equivalent_to(Q3, True); M [-1 -1 0] [ 1 1 1] [-1 0 0] >>> Q1(M) == Q3 True - sage: Q = DiagonalQuadraticForm(ZZ, [1, -1]) sage: Q.is_globally_equivalent_to(Q) # needs sage.libs.pari Traceback (most recent call last): ... ValueError: not a definite form in QuadraticForm.is_globally_equivalent_to() - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1), -Integer(1)]) >>> Q.is_globally_equivalent_to(Q) # needs sage.libs.pari Traceback (most recent call last): ... ValueError: not a definite form in QuadraticForm.is_globally_equivalent_to() - ALGORITHM: this uses the PARI function pari:qfisom, implementing an algorithm by Plesken and Souvignier. 
 - is_hyperbolic(p)[source]¶
- Check if the quadratic form is a sum of hyperbolic planes over the \(p\)-adic numbers \(\QQ_p\) or over the real numbers \(\RR\). - REFERENCES: - This criterion follows from Cassels’s “Rational Quadratic Forms”: - local invariants for hyperbolic plane (Lemma 2.4, p58) 
- direct sum formulas (Lemma 2.3, p58) 
 - INPUT: - p– a prime number > 0 or \(-1\) for the infinite place
 - OUTPUT: boolean - EXAMPLES: - sage: # needs sage.libs.pari sage: Q = DiagonalQuadraticForm(ZZ, [1,1]) sage: Q.is_hyperbolic(-1) False sage: Q.is_hyperbolic(2) False sage: Q.is_hyperbolic(3) False sage: Q.is_hyperbolic(5) # Here -1 is a square, so it's true. True sage: Q.is_hyperbolic(7) False sage: Q.is_hyperbolic(13) # Here -1 is a square, so it's true. True - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]) >>> Q.is_hyperbolic(-Integer(1)) False >>> Q.is_hyperbolic(Integer(2)) False >>> Q.is_hyperbolic(Integer(3)) False >>> Q.is_hyperbolic(Integer(5)) # Here -1 is a square, so it's true. True >>> Q.is_hyperbolic(Integer(7)) False >>> Q.is_hyperbolic(Integer(13)) # Here -1 is a square, so it's true. True 
 - is_indefinite()[source]¶
- Determines if the given quadratic form is indefinite. - Note - A degenerate form is considered neither definite nor indefinite. - Note - The zero-dimensional form is not considered indefinite. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [-1,-3,-5]) sage: Q.is_indefinite() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [-Integer(1),-Integer(3),-Integer(5)]) >>> Q.is_indefinite() False - sage: Q = DiagonalQuadraticForm(ZZ, [1,-3,5]) sage: Q.is_indefinite() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(3),Integer(5)]) >>> Q.is_indefinite() True 
 - is_isotropic(p)[source]¶
- Check if \(Q\) is isotropic over the \(p\)-adic numbers \(\QQ_p\) or \(\RR\). - INPUT: - p– a prime number > 0 or \(-1\) for the infinite place
 - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1]) sage: Q.is_isotropic(2) # needs sage.libs.pari False sage: Q.is_isotropic(3) # needs sage.libs.pari False sage: Q.is_isotropic(5) # needs sage.libs.pari True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]) >>> Q.is_isotropic(Integer(2)) # needs sage.libs.pari False >>> Q.is_isotropic(Integer(3)) # needs sage.libs.pari False >>> Q.is_isotropic(Integer(5)) # needs sage.libs.pari True - sage: Q = DiagonalQuadraticForm(ZZ, [1,-1]) sage: Q.is_isotropic(2) # needs sage.libs.pari True sage: Q.is_isotropic(3) # needs sage.libs.pari True sage: Q.is_isotropic(5) # needs sage.libs.pari True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(1)]) >>> Q.is_isotropic(Integer(2)) # needs sage.libs.pari True >>> Q.is_isotropic(Integer(3)) # needs sage.libs.pari True >>> Q.is_isotropic(Integer(5)) # needs sage.libs.pari True - sage: [DiagonalQuadraticForm(ZZ, # needs sage.libs.pari ....: [1, -least_quadratic_nonresidue(p)]).is_isotropic(p) ....: for p in prime_range(3, 30)] [False, False, False, False, False, False, False, False, False] - >>> from sage.all import * >>> [DiagonalQuadraticForm(ZZ, # needs sage.libs.pari ... [Integer(1), -least_quadratic_nonresidue(p)]).is_isotropic(p) ... for p in prime_range(Integer(3), Integer(30))] [False, False, False, False, False, False, False, False, False] - sage: [DiagonalQuadraticForm(ZZ, [1, -least_quadratic_nonresidue(p), # needs sage.libs.pari ....: p, -p*least_quadratic_nonresidue(p)]).is_isotropic(p) ....: for p in prime_range(3, 30)] [False, False, False, False, False, False, False, False, False] - >>> from sage.all import * >>> [DiagonalQuadraticForm(ZZ, [Integer(1), -least_quadratic_nonresidue(p), # needs sage.libs.pari ... p, -p*least_quadratic_nonresidue(p)]).is_isotropic(p) ... for p in prime_range(Integer(3), Integer(30))] [False, False, False, False, False, False, False, False, False] 
 - is_locally_equivalent_to(other, check_primes_only=False, force_jordan_equivalence_test=False)[source]¶
- Determine if the current quadratic form (defined over \(\ZZ\)) is locally equivalent to the given form over the real numbers and the \(p\)-adic integers for every prime \(p\). - This works by comparing the local Jordan decompositions at every prime, and the dimension and signature at the real place. - INPUT: - other– a- QuadraticForm
 - OUTPUT: boolean - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q2 = QuadraticForm(ZZ, 3, [2, 1, 2, 2, 1, 3]) sage: Q1.is_globally_equivalent_to(Q2) # needs sage.libs.pari False sage: Q1.is_locally_equivalent_to(Q2) # needs sage.libs.pari True - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q2 = QuadraticForm(ZZ, Integer(3), [Integer(2), Integer(1), Integer(2), Integer(2), Integer(1), Integer(3)]) >>> Q1.is_globally_equivalent_to(Q2) # needs sage.libs.pari False >>> Q1.is_locally_equivalent_to(Q2) # needs sage.libs.pari True 
 - is_locally_represented_number(m)[source]¶
- Determine if the rational number \(m\) is locally represented by the quadratic form. - INPUT: - m– integer
 - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.is_locally_represented_number(2) True sage: Q.is_locally_represented_number(7) False sage: Q.is_locally_represented_number(-1) False sage: Q.is_locally_represented_number(28) False sage: Q.is_locally_represented_number(0) True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_represented_number(Integer(2)) True >>> Q.is_locally_represented_number(Integer(7)) False >>> Q.is_locally_represented_number(-Integer(1)) False >>> Q.is_locally_represented_number(Integer(28)) False >>> Q.is_locally_represented_number(Integer(0)) True 
 - is_locally_represented_number_at_place(m, p)[source]¶
- Determine if the rational number \(m\) is locally represented by the quadratic form at the (possibly infinite) prime \(p\). - INPUT: - m– integer
- p– a prime number > 0 or ‘infinity’
 - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.is_locally_represented_number_at_place(7, infinity) True sage: Q.is_locally_represented_number_at_place(7, 2) False sage: Q.is_locally_represented_number_at_place(7, 3) True sage: Q.is_locally_represented_number_at_place(7, 5) True sage: Q.is_locally_represented_number_at_place(-1, infinity) False sage: Q.is_locally_represented_number_at_place(-1, 2) False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_represented_number_at_place(Integer(7), infinity) True >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(2)) False >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(3)) True >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(5)) True >>> Q.is_locally_represented_number_at_place(-Integer(1), infinity) False >>> Q.is_locally_represented_number_at_place(-Integer(1), Integer(2)) False - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,-1]) sage: Q.is_locally_represented_number_at_place(7, infinity) # long time (8.5 s) True sage: Q.is_locally_represented_number_at_place(7, 2) # long time True sage: Q.is_locally_represented_number_at_place(7, 3) # long time True sage: Q.is_locally_represented_number_at_place(7, 5) # long time True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),-Integer(1)]) >>> Q.is_locally_represented_number_at_place(Integer(7), infinity) # long time (8.5 s) True >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(2)) # long time True >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(3)) # long time True >>> Q.is_locally_represented_number_at_place(Integer(7), Integer(5)) # long time True 
 - is_locally_universal_at_all_places()[source]¶
- Determine if the quadratic form represents \(\ZZ_p\) for all finite/non-archimedean primes, and represents all real numbers. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.is_locally_universal_at_all_places() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.is_locally_universal_at_all_places() False - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.is_locally_universal_at_all_places() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_universal_at_all_places() False - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,-1]) sage: Q.is_locally_universal_at_all_places() # long time (8.5 s) True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),-Integer(1)]) >>> Q.is_locally_universal_at_all_places() # long time (8.5 s) True 
 - is_locally_universal_at_all_primes()[source]¶
- Determine if the quadratic form represents \(\ZZ_p\) for all finite/non-archimedean primes. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.is_locally_universal_at_all_primes() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.is_locally_universal_at_all_primes() True - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.is_locally_universal_at_all_primes() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_universal_at_all_primes() True - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.is_locally_universal_at_all_primes() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_universal_at_all_primes() False 
 - is_locally_universal_at_prime(p)[source]¶
- Determine if the (integer-valued/rational) quadratic form represents all of \(\ZZ_p\). - INPUT: - p– a positive prime number or “infinity”
 - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.is_locally_universal_at_prime(2) True sage: Q.is_locally_universal_at_prime(3) True sage: Q.is_locally_universal_at_prime(5) True sage: Q.is_locally_universal_at_prime(infinity) False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.is_locally_universal_at_prime(Integer(2)) True >>> Q.is_locally_universal_at_prime(Integer(3)) True >>> Q.is_locally_universal_at_prime(Integer(5)) True >>> Q.is_locally_universal_at_prime(infinity) False - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.is_locally_universal_at_prime(2) False sage: Q.is_locally_universal_at_prime(3) True sage: Q.is_locally_universal_at_prime(5) True sage: Q.is_locally_universal_at_prime(infinity) False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.is_locally_universal_at_prime(Integer(2)) False >>> Q.is_locally_universal_at_prime(Integer(3)) True >>> Q.is_locally_universal_at_prime(Integer(5)) True >>> Q.is_locally_universal_at_prime(infinity) False - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,-1]) sage: Q.is_locally_universal_at_prime(infinity) True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),-Integer(1)]) >>> Q.is_locally_universal_at_prime(infinity) True 
 - is_negative_definite()[source]¶
- Determines if the given quadratic form is negative-definite. - Note - A degenerate form is considered neither definite nor indefinite. - Note - The zero-dimensional form is considered both positive definite and negative definite. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [-1,-3,-5]) sage: Q.is_negative_definite() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [-Integer(1),-Integer(3),-Integer(5)]) >>> Q.is_negative_definite() True - sage: Q = DiagonalQuadraticForm(ZZ, [1,-3,5]) sage: Q.is_negative_definite() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(3),Integer(5)]) >>> Q.is_negative_definite() False 
 - is_odd(allow_rescaling_flag=True)[source]¶
- Return true iff after rescaling by some appropriate factor, the form represents some odd integers. For more details, see - parity().- Requires that \(Q\) is defined over \(\ZZ\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1, 0, 1]) sage: Q.is_odd() True sage: Q = QuadraticForm(ZZ, 2, [1, 1, 1]) sage: Q.is_odd() False - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1), Integer(0), Integer(1)]) >>> Q.is_odd() True >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1), Integer(1), Integer(1)]) >>> Q.is_odd() False 
 - is_positive_definite()[source]¶
- Determines if the given quadratic form is positive-definite. - Note - A degenerate form is considered neither definite nor indefinite. - Note - The zero-dimensional form is considered both positive definite and negative definite. - OUTPUT: boolean - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5]) sage: Q.is_positive_definite() True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5)]) >>> Q.is_positive_definite() True - sage: Q = DiagonalQuadraticForm(ZZ, [1,-3,5]) sage: Q.is_positive_definite() False - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),-Integer(3),Integer(5)]) >>> Q.is_positive_definite() False 
 - is_primitive()[source]¶
- Determine if the given integer-valued form is primitive. - This means not an integer (\(> 1\)) multiple of another integer-valued quadratic form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [2,3,4]) sage: Q.is_primitive() True sage: Q = QuadraticForm(ZZ, 2, [2,4,8]) sage: Q.is_primitive() False - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2),Integer(3),Integer(4)]) >>> Q.is_primitive() True >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2),Integer(4),Integer(8)]) >>> Q.is_primitive() False 
 - is_rationally_isometric(other, return_matrix=False)[source]¶
- Determine if two regular quadratic forms over a number field are isometric. - INPUT: - other– a quadratic form over a number field
- return_matrix– boolean (default:- False); return the transformation matrix instead of a boolean; this is currently only implemented for forms over- QQ
 - OUTPUT: - if - return_matrixis- False: a boolean
- if - return_matrixis- True: either- Falseor the transformation matrix
 - EXAMPLES: - sage: V = DiagonalQuadraticForm(QQ, [1, 1, 2]) sage: W = DiagonalQuadraticForm(QQ, [2, 2, 2]) sage: V.is_rationally_isometric(W) # needs sage.libs.pari True - >>> from sage.all import * >>> V = DiagonalQuadraticForm(QQ, [Integer(1), Integer(1), Integer(2)]) >>> W = DiagonalQuadraticForm(QQ, [Integer(2), Integer(2), Integer(2)]) >>> V.is_rationally_isometric(W) # needs sage.libs.pari True - sage: # needs sage.rings.number_field sage: x = polygen(ZZ, 'x') sage: K.<a> = NumberField(x^2 - 3) sage: V = QuadraticForm(K, 4, [1, 0, 0, 0, 2*a, 0, 0, a, 0, 2]); V Quadratic form in 4 variables over Number Field in a with defining polynomial x^2 - 3 with coefficients: [ 1 0 0 0 ] [ * 2*a 0 0 ] [ * * a 0 ] [ * * * 2 ] sage: W = QuadraticForm(K, 4, [1, 2*a, 4, 6, 3, 10, 2, 1, 2, 5]); W Quadratic form in 4 variables over Number Field in a with defining polynomial x^2 - 3 with coefficients: [ 1 2*a 4 6 ] [ * 3 10 2 ] [ * * 1 2 ] [ * * * 5 ] sage: V.is_rationally_isometric(W) False - >>> from sage.all import * >>> # needs sage.rings.number_field >>> x = polygen(ZZ, 'x') >>> K = NumberField(x**Integer(2) - Integer(3), names=('a',)); (a,) = K._first_ngens(1) >>> V = QuadraticForm(K, Integer(4), [Integer(1), Integer(0), Integer(0), Integer(0), Integer(2)*a, Integer(0), Integer(0), a, Integer(0), Integer(2)]); V Quadratic form in 4 variables over Number Field in a with defining polynomial x^2 - 3 with coefficients: [ 1 0 0 0 ] [ * 2*a 0 0 ] [ * * a 0 ] [ * * * 2 ] >>> W = QuadraticForm(K, Integer(4), [Integer(1), Integer(2)*a, Integer(4), Integer(6), Integer(3), Integer(10), Integer(2), Integer(1), Integer(2), Integer(5)]); W Quadratic form in 4 variables over Number Field in a with defining polynomial x^2 - 3 with coefficients: [ 1 2*a 4 6 ] [ * 3 10 2 ] [ * * 1 2 ] [ * * * 5 ] >>> V.is_rationally_isometric(W) False - sage: # needs sage.rings.number_field sage: K.<a> = NumberField(x^4 + 2*x + 6) sage: V = DiagonalQuadraticForm(K, [a, 2, 3, 2, 1]); V Quadratic form in 5 variables over Number Field in a with defining polynomial x^4 + 2*x + 6 with coefficients: [ a 0 0 0 0 ] [ * 2 0 0 0 ] [ * * 3 0 0 ] [ * * * 2 0 ] [ * * * * 1 ] sage: W = DiagonalQuadraticForm(K, [a, a, a, 2, 1]); W Quadratic form in 5 variables over Number Field in a with defining polynomial x^4 + 2*x + 6 with coefficients: [ a 0 0 0 0 ] [ * a 0 0 0 ] [ * * a 0 0 ] [ * * * 2 0 ] [ * * * * 1 ] sage: V.is_rationally_isometric(W) False - >>> from sage.all import * >>> # needs sage.rings.number_field >>> K = NumberField(x**Integer(4) + Integer(2)*x + Integer(6), names=('a',)); (a,) = K._first_ngens(1) >>> V = DiagonalQuadraticForm(K, [a, Integer(2), Integer(3), Integer(2), Integer(1)]); V Quadratic form in 5 variables over Number Field in a with defining polynomial x^4 + 2*x + 6 with coefficients: [ a 0 0 0 0 ] [ * 2 0 0 0 ] [ * * 3 0 0 ] [ * * * 2 0 ] [ * * * * 1 ] >>> W = DiagonalQuadraticForm(K, [a, a, a, Integer(2), Integer(1)]); W Quadratic form in 5 variables over Number Field in a with defining polynomial x^4 + 2*x + 6 with coefficients: [ a 0 0 0 0 ] [ * a 0 0 0 ] [ * * a 0 0 ] [ * * * 2 0 ] [ * * * * 1 ] >>> V.is_rationally_isometric(W) False - sage: # needs sage.rings.number_field sage: K.<a> = NumberField(x^2 - 3) sage: V = DiagonalQuadraticForm(K, [-1, a, -2*a]) sage: W = DiagonalQuadraticForm(K, [-1, -a, 2*a]) sage: V.is_rationally_isometric(W) True sage: # needs sage.rings.number_field sage: V = DiagonalQuadraticForm(QQ, [1, 1, 2]) sage: W = DiagonalQuadraticForm(QQ, [2, 2, 2]) sage: T = V.is_rationally_isometric(W, True); T [ 0 0 1] [-1/2 -1/2 0] [ 1/2 -1/2 0] sage: V.Gram_matrix() == T.transpose() * W.Gram_matrix() * T True sage: T = W.is_rationally_isometric(V, True); T # needs sage.rings.number_field [ 0 -1 1] [ 0 -1 -1] [ 1 0 0] sage: W.Gram_matrix() == T.T * V.Gram_matrix() * T # needs sage.rings.number_field True - >>> from sage.all import * >>> # needs sage.rings.number_field >>> K = NumberField(x**Integer(2) - Integer(3), names=('a',)); (a,) = K._first_ngens(1) >>> V = DiagonalQuadraticForm(K, [-Integer(1), a, -Integer(2)*a]) >>> W = DiagonalQuadraticForm(K, [-Integer(1), -a, Integer(2)*a]) >>> V.is_rationally_isometric(W) True >>> # needs sage.rings.number_field >>> V = DiagonalQuadraticForm(QQ, [Integer(1), Integer(1), Integer(2)]) >>> W = DiagonalQuadraticForm(QQ, [Integer(2), Integer(2), Integer(2)]) >>> T = V.is_rationally_isometric(W, True); T [ 0 0 1] [-1/2 -1/2 0] [ 1/2 -1/2 0] >>> V.Gram_matrix() == T.transpose() * W.Gram_matrix() * T True >>> T = W.is_rationally_isometric(V, True); T # needs sage.rings.number_field [ 0 -1 1] [ 0 -1 -1] [ 1 0 0] >>> W.Gram_matrix() == T.T * V.Gram_matrix() * T # needs sage.rings.number_field True - sage: L = QuadraticForm(QQ, 3, [2, 2, 0, 2, 2, 5]) sage: M = QuadraticForm(QQ, 3, [2, 2, 0, 3, 2, 3]) sage: L.is_rationally_isometric(M, True) # needs sage.libs.pari False - >>> from sage.all import * >>> L = QuadraticForm(QQ, Integer(3), [Integer(2), Integer(2), Integer(0), Integer(2), Integer(2), Integer(5)]) >>> M = QuadraticForm(QQ, Integer(3), [Integer(2), Integer(2), Integer(0), Integer(3), Integer(2), Integer(3)]) >>> L.is_rationally_isometric(M, True) # needs sage.libs.pari False - sage: A = DiagonalQuadraticForm(QQ, [1, 5]) sage: B = QuadraticForm(QQ, 2, [1, 12, 81]) sage: T = A.is_rationally_isometric(B, True); T # needs sage.libs.pari [ 1 -2] [ 0 1/3] sage: A.Gram_matrix() == T.T * B.Gram_matrix() * T # needs sage.libs.pari True - >>> from sage.all import * >>> A = DiagonalQuadraticForm(QQ, [Integer(1), Integer(5)]) >>> B = QuadraticForm(QQ, Integer(2), [Integer(1), Integer(12), Integer(81)]) >>> T = A.is_rationally_isometric(B, True); T # needs sage.libs.pari [ 1 -2] [ 0 1/3] >>> A.Gram_matrix() == T.T * B.Gram_matrix() * T # needs sage.libs.pari True - sage: C = DiagonalQuadraticForm(QQ, [1, 5, 9]) sage: D = DiagonalQuadraticForm(QQ, [6, 30, 1]) sage: T = C.is_rationally_isometric(D, True); T # needs sage.libs.pari [ 0 -5/6 1/2] [ 0 1/6 1/2] [ -1 0 0] sage: C.Gram_matrix() == T.T * D.Gram_matrix() * T # needs sage.libs.pari True - >>> from sage.all import * >>> C = DiagonalQuadraticForm(QQ, [Integer(1), Integer(5), Integer(9)]) >>> D = DiagonalQuadraticForm(QQ, [Integer(6), Integer(30), Integer(1)]) >>> T = C.is_rationally_isometric(D, True); T # needs sage.libs.pari [ 0 -5/6 1/2] [ 0 1/6 1/2] [ -1 0 0] >>> C.Gram_matrix() == T.T * D.Gram_matrix() * T # needs sage.libs.pari True - sage: E = DiagonalQuadraticForm(QQ, [1, 1]) sage: F = QuadraticForm(QQ, 2, [17, 94, 130]) sage: T = F.is_rationally_isometric(E, True); T # needs sage.libs.pari [ -4 -189/17] [ -1 -43/17] sage: F.Gram_matrix() == T.T * E.Gram_matrix() * T # needs sage.libs.pari True - >>> from sage.all import * >>> E = DiagonalQuadraticForm(QQ, [Integer(1), Integer(1)]) >>> F = QuadraticForm(QQ, Integer(2), [Integer(17), Integer(94), Integer(130)]) >>> T = F.is_rationally_isometric(E, True); T # needs sage.libs.pari [ -4 -189/17] [ -1 -43/17] >>> F.Gram_matrix() == T.T * E.Gram_matrix() * T # needs sage.libs.pari True 
 - is_zero(v, p=0)[source]¶
- Determine if the vector \(v\) is on the conic \(Q(x) = 0\) (mod \(p\)). - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q1.is_zero([0,1,0], 2) True sage: Q1.is_zero([1,1,1], 2) True sage: Q1.is_zero([1,1,0], 2) False - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q1.is_zero([Integer(0),Integer(1),Integer(0)], Integer(2)) True >>> Q1.is_zero([Integer(1),Integer(1),Integer(1)], Integer(2)) True >>> Q1.is_zero([Integer(1),Integer(1),Integer(0)], Integer(2)) False 
 - is_zero_nonsingular(v, p=0)[source]¶
- Determine if the vector \(v\) is on the conic \(Q(x) = 0\) (mod \(p\)), and that this point is non-singular point of the conic. - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q1.is_zero_nonsingular([1,1,1], 2) True sage: Q1.is_zero([1, 19, 2], 37) True sage: Q1.is_zero_nonsingular([1, 19, 2], 37) False - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q1.is_zero_nonsingular([Integer(1),Integer(1),Integer(1)], Integer(2)) True >>> Q1.is_zero([Integer(1), Integer(19), Integer(2)], Integer(37)) True >>> Q1.is_zero_nonsingular([Integer(1), Integer(19), Integer(2)], Integer(37)) False 
 - is_zero_singular(v, p=0)[source]¶
- Determine if the vector \(v\) is on the conic \(Q(x) = 0\) (mod \(p\)), and that this point is singular point of the conic. - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 0, -1, 2, -1, 5]) sage: Q1.is_zero([1,1,1], 2) True sage: Q1.is_zero_singular([1,1,1], 2) False sage: Q1.is_zero_singular([1, 19, 2], 37) True - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), -Integer(1), Integer(2), -Integer(1), Integer(5)]) >>> Q1.is_zero([Integer(1),Integer(1),Integer(1)], Integer(2)) True >>> Q1.is_zero_singular([Integer(1),Integer(1),Integer(1)], Integer(2)) False >>> Q1.is_zero_singular([Integer(1), Integer(19), Integer(2)], Integer(37)) True 
 - jordan_blocks_by_scale_and_unimodular(p, safe_flag=True)[source]¶
- Return a list of pairs \((s_i, L_i)\) where \(L_i\) is a maximal \(p^{s_i}\)-unimodular Jordan component which is further decomposed into block diagonals of block size \(\le 2\). - For each \(L_i\) the \(2 \times 2\) blocks are listed after the \(1 \times 1\) blocks (which follows from the convention of the - local_normal_form()method).- Note - The decomposition of each \(L_i\) into smaller blocks is not unique! - The - safe_flagargument allows us to select whether we want a copy of the output, or the original output. By default- safe_flag = True, so we return a copy of the cached information. If this is set to- False, then the routine is much faster but the return values are vulnerable to being corrupted by the user.- INPUT: - p– a prime number > 0
 - OUTPUT: - A list of pairs \((s_i, L_i)\) where: - \(s_i\) is an integer, 
- \(L_i\) is a block-diagonal unimodular quadratic form over \(\ZZ_p\). 
 - Note - These forms \(L_i\) are defined over the \(p\)-adic integers, but by a matrix over \(\ZZ\) (or \(\QQ\)?). - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,9,5,7]) sage: Q.jordan_blocks_by_scale_and_unimodular(3) [(0, Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 5 0 ] [ * * 7 ]), (2, Quadratic form in 1 variables over Integer Ring with coefficients: [ 1 ])] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(9),Integer(5),Integer(7)]) >>> Q.jordan_blocks_by_scale_and_unimodular(Integer(3)) [(0, Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 5 0 ] [ * * 7 ]), (2, Quadratic form in 1 variables over Integer Ring with coefficients: [ 1 ])] - sage: Q2 = QuadraticForm(ZZ, 2, [1,1,1]) sage: Q2.jordan_blocks_by_scale_and_unimodular(2) [(-1, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ])] sage: Q = Q2 + Q2.scale_by_factor(2) sage: Q.jordan_blocks_by_scale_and_unimodular(2) [(-1, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ]), (0, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ])] - >>> from sage.all import * >>> Q2 = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(1),Integer(1)]) >>> Q2.jordan_blocks_by_scale_and_unimodular(Integer(2)) [(-1, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ])] >>> Q = Q2 + Q2.scale_by_factor(Integer(2)) >>> Q.jordan_blocks_by_scale_and_unimodular(Integer(2)) [(-1, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ]), (0, Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 2 ] [ * 2 ])] 
 - jordan_blocks_in_unimodular_list_by_scale_power(p)[source]¶
- Return a list of Jordan components, whose component at index \(i\) should be scaled by the factor \(p^i\). - This is only defined for integer-valued quadratic forms (i.e., forms with base ring \(\ZZ\)), and the indexing only works correctly for \(p=2\) when the form has an integer Gram matrix. - INPUT: - self– a quadratic form over \(\ZZ\), which has integer Gram matrix if \(p = 2\)
- p– a prime number > 0
 - OUTPUT: list of \(p\)-unimodular quadratic forms - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [2, -2, 0, 3, -5, 4]) sage: Q.jordan_blocks_in_unimodular_list_by_scale_power(2) Traceback (most recent call last): ... TypeError: the given quadratic form has a Jordan component with a negative scale exponent sage: Q.scale_by_factor(2).jordan_blocks_in_unimodular_list_by_scale_power(2) [Quadratic form in 2 variables over Integer Ring with coefficients: [ 0 2 ] [ * 0 ], Quadratic form in 0 variables over Integer Ring with coefficients: , Quadratic form in 1 variables over Integer Ring with coefficients: [ 345 ]] sage: Q.jordan_blocks_in_unimodular_list_by_scale_power(3) [Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 0 ] [ * 10 ], Quadratic form in 1 variables over Integer Ring with coefficients: [ 2 ]] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(2), Integer(0), Integer(3), -Integer(5), Integer(4)]) >>> Q.jordan_blocks_in_unimodular_list_by_scale_power(Integer(2)) Traceback (most recent call last): ... TypeError: the given quadratic form has a Jordan component with a negative scale exponent >>> Q.scale_by_factor(Integer(2)).jordan_blocks_in_unimodular_list_by_scale_power(Integer(2)) [Quadratic form in 2 variables over Integer Ring with coefficients: [ 0 2 ] [ * 0 ], Quadratic form in 0 variables over Integer Ring with coefficients: , Quadratic form in 1 variables over Integer Ring with coefficients: [ 345 ]] >>> Q.jordan_blocks_in_unimodular_list_by_scale_power(Integer(3)) [Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 0 ] [ * 10 ], Quadratic form in 1 variables over Integer Ring with coefficients: [ 2 ]] 
 - level()[source]¶
- Determines the level of the quadratic form over a PID, which is a generator for the smallest ideal \(N\) of \(R\) such that \(N\cdot (\) the matrix of \(2*Q\) \()^{(-1)}\) is in \(R\) with diagonal in \(2R\). - Over \(\ZZ\) this returns a nonnegative number. - (Caveat: This always returns the unit ideal when working over a field!) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, range(1,4)) sage: Q.level() 8 sage: Q1 = QuadraticForm(QQ, 2, range(1,4)) sage: Q1.level() # random UserWarning: Warning -- The level of a quadratic form over a field is always 1. Do you really want to do this?!? 1 sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.level() 420 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), range(Integer(1),Integer(4))) >>> Q.level() 8 >>> Q1 = QuadraticForm(QQ, Integer(2), range(Integer(1),Integer(4))) >>> Q1.level() # random UserWarning: Warning -- The level of a quadratic form over a field is always 1. Do you really want to do this?!? 1 >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.level() 420 
 - level__Tornaria()[source]¶
- Return the level of the quadratic form. - This is defined as - level(\(B\)) for even dimension, 
- level(\(B\))/4 for odd dimension, 
 - where \(2Q(x) = x^t\cdot B\cdot x\). - This agrees with the usual level for even dimension. - EXAMPLES: - sage: DiagonalQuadraticForm(ZZ, [1]).level__Tornaria() 1 sage: DiagonalQuadraticForm(ZZ, [1,1]).level__Tornaria() 4 sage: DiagonalQuadraticForm(ZZ, [1,1,1]).level__Tornaria() 1 sage: DiagonalQuadraticForm(ZZ, [1,1,1,1]).level__Tornaria() 4 - >>> from sage.all import * >>> DiagonalQuadraticForm(ZZ, [Integer(1)]).level__Tornaria() 1 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]).level__Tornaria() 4 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]).level__Tornaria() 1 >>> DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]).level__Tornaria() 4 
 - level_ideal()[source]¶
- Determine the level of the quadratic form (over \(R\)), which is the smallest ideal \(N\) of \(R\) such that \(N \cdot (\) the matrix of \(2Q\) \()^{(-1)}\) is in \(R\) with diagonal in \(2R\). (Caveat: This always returns the principal ideal when working over a field!) - Warning - This only works over a PID ring of integers for now! (Waiting for Sage fractional ideal support.) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, range(1,4)) sage: Q.level_ideal() Principal ideal (8) of Integer Ring sage: Q1 = QuadraticForm(QQ, 2, range(1,4)) sage: Q1.level_ideal() Principal ideal (1) of Rational Field sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.level_ideal() Principal ideal (420) of Integer Ring - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), range(Integer(1),Integer(4))) >>> Q.level_ideal() Principal ideal (8) of Integer Ring >>> Q1 = QuadraticForm(QQ, Integer(2), range(Integer(1),Integer(4))) >>> Q1.level_ideal() Principal ideal (1) of Rational Field >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.level_ideal() Principal ideal (420) of Integer Ring 
 - list_external_initializations()[source]¶
- Return a list of the fields which were set externally at creation, and not created through the usual - QuadraticFormmethods. These fields are as good as the external process that made them, and are thus not guaranteed to be correct.- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,0,5]) sage: Q.list_external_initializations() [] sage: # needs sage.libs.pari sage: T = Q.theta_series() sage: Q.list_external_initializations() [] sage: Q = QuadraticForm(ZZ, 2, [1,0,5], unsafe_initialization=False, ....: number_of_automorphisms=3, determinant=0) sage: Q.list_external_initializations() [] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(0),Integer(5)]) >>> Q.list_external_initializations() [] >>> # needs sage.libs.pari >>> T = Q.theta_series() >>> Q.list_external_initializations() [] >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(0),Integer(5)], unsafe_initialization=False, ... number_of_automorphisms=Integer(3), determinant=Integer(0)) >>> Q.list_external_initializations() [] - sage: # needs sage.libs.pari sage: Q = QuadraticForm(ZZ, 2, [1,0,5], unsafe_initialization=False, ....: number_of_automorphisms=3, determinant=0) sage: Q.list_external_initializations() [] sage: Q = QuadraticForm(ZZ, 2, [1,0,5], unsafe_initialization=True, ....: number_of_automorphisms=3, determinant=0) sage: Q.list_external_initializations() ['number_of_automorphisms', 'determinant'] - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(0),Integer(5)], unsafe_initialization=False, ... number_of_automorphisms=Integer(3), determinant=Integer(0)) >>> Q.list_external_initializations() [] >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(0),Integer(5)], unsafe_initialization=True, ... number_of_automorphisms=Integer(3), determinant=Integer(0)) >>> Q.list_external_initializations() ['number_of_automorphisms', 'determinant'] 
 - lll()[source]¶
- Return an LLL-reduced form of \(Q\) (using PARI). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(1,11)) sage: Q.is_definite() True sage: Q.lll() # needs sage.libs.pari Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 1 0 ] [ * 4 3 3 ] [ * * 6 3 ] [ * * * 6 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(1),Integer(11))) >>> Q.is_definite() True >>> Q.lll() # needs sage.libs.pari Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 1 0 ] [ * 4 3 3 ] [ * * 6 3 ] [ * * * 6 ] 
 - local_badII_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the Bad-type II local density of \(Q\) representing \(m\) at \(p\). (Assuming that \(p > 2\) and \(Q\) is given in local diagonal form.) - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_badII_density_congruence(2, 1, None, None) 0 sage: Q.local_badII_density_congruence(2, 2, None, None) 0 sage: Q.local_badII_density_congruence(2, 4, None, None) 0 sage: Q.local_badII_density_congruence(3, 1, None, None) 0 sage: Q.local_badII_density_congruence(3, 6, None, None) 0 sage: Q.local_badII_density_congruence(3, 9, None, None) 0 sage: Q.local_badII_density_congruence(3, 27, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_badII_density_congruence(Integer(2), Integer(1), None, None) 0 >>> Q.local_badII_density_congruence(Integer(2), Integer(2), None, None) 0 >>> Q.local_badII_density_congruence(Integer(2), Integer(4), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(9), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(27), None, None) 0 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,3,9,9]) sage: Q.local_badII_density_congruence(3, 1, None, None) 0 sage: Q.local_badII_density_congruence(3, 3, None, None) 0 sage: Q.local_badII_density_congruence(3, 6, None, None) 0 sage: Q.local_badII_density_congruence(3, 9, None, None) 4/27 sage: Q.local_badII_density_congruence(3, 18, None, None) 4/9 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(9),Integer(9)]) >>> Q.local_badII_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(3), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_badII_density_congruence(Integer(3), Integer(9), None, None) 4/27 >>> Q.local_badII_density_congruence(Integer(3), Integer(18), None, None) 4/9 
 - local_badI_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the Bad-type I local density of \(Q\) representing \(m\) at \(p\). (Assuming that \(p > 2\) and \(Q\) is given in local diagonal form.) - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_badI_density_congruence(2, 1, None, None) 0 sage: Q.local_badI_density_congruence(2, 2, None, None) 1 sage: Q.local_badI_density_congruence(2, 4, None, None) 0 sage: Q.local_badI_density_congruence(3, 1, None, None) 0 sage: Q.local_badI_density_congruence(3, 6, None, None) 0 sage: Q.local_badI_density_congruence(3, 9, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_badI_density_congruence(Integer(2), Integer(1), None, None) 0 >>> Q.local_badI_density_congruence(Integer(2), Integer(2), None, None) 1 >>> Q.local_badI_density_congruence(Integer(2), Integer(4), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(9), None, None) 0 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_badI_density_congruence(2, 1, None, None) 0 sage: Q.local_badI_density_congruence(2, 2, None, None) 0 sage: Q.local_badI_density_congruence(2, 4, None, None) 0 sage: Q.local_badI_density_congruence(3, 2, None, None) 0 sage: Q.local_badI_density_congruence(3, 6, None, None) 0 sage: Q.local_badI_density_congruence(3, 9, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_badI_density_congruence(Integer(2), Integer(1), None, None) 0 >>> Q.local_badI_density_congruence(Integer(2), Integer(2), None, None) 0 >>> Q.local_badI_density_congruence(Integer(2), Integer(4), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(2), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(9), None, None) 0 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,3,9]) sage: Q.local_badI_density_congruence(3, 1, None, None) 0 sage: Q.local_badI_density_congruence(3, 3, None, None) 4/3 sage: Q.local_badI_density_congruence(3, 6, None, None) 4/3 sage: Q.local_badI_density_congruence(3, 9, None, None) 0 sage: Q.local_badI_density_congruence(3, 18, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(9)]) >>> Q.local_badI_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(3), None, None) 4/3 >>> Q.local_badI_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_badI_density_congruence(Integer(3), Integer(9), None, None) 0 >>> Q.local_badI_density_congruence(Integer(3), Integer(18), None, None) 0 
 - local_bad_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the Bad-type local density of \(Q\) representing \(m\) at \(p\), allowing certain congruence conditions mod \(p\). - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_bad_density_congruence(2, 1, None, None) 0 sage: Q.local_bad_density_congruence(2, 2, None, None) 1 sage: Q.local_bad_density_congruence(2, 4, None, None) 0 sage: Q.local_bad_density_congruence(3, 1, None, None) 0 sage: Q.local_bad_density_congruence(3, 6, None, None) 0 sage: Q.local_bad_density_congruence(3, 9, None, None) 0 sage: Q.local_bad_density_congruence(3, 27, None, None) 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_bad_density_congruence(Integer(2), Integer(1), None, None) 0 >>> Q.local_bad_density_congruence(Integer(2), Integer(2), None, None) 1 >>> Q.local_bad_density_congruence(Integer(2), Integer(4), None, None) 0 >>> Q.local_bad_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_bad_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_bad_density_congruence(Integer(3), Integer(9), None, None) 0 >>> Q.local_bad_density_congruence(Integer(3), Integer(27), None, None) 0 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,3,9,9]) sage: Q.local_bad_density_congruence(3, 1, None, None) 0 sage: Q.local_bad_density_congruence(3, 3, None, None) 4/3 sage: Q.local_bad_density_congruence(3, 6, None, None) 4/3 sage: Q.local_bad_density_congruence(3, 9, None, None) 4/27 sage: Q.local_bad_density_congruence(3, 18, None, None) 4/9 sage: Q.local_bad_density_congruence(3, 27, None, None) 8/27 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(9),Integer(9)]) >>> Q.local_bad_density_congruence(Integer(3), Integer(1), None, None) 0 >>> Q.local_bad_density_congruence(Integer(3), Integer(3), None, None) 4/3 >>> Q.local_bad_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_bad_density_congruence(Integer(3), Integer(9), None, None) 4/27 >>> Q.local_bad_density_congruence(Integer(3), Integer(18), None, None) 4/9 >>> Q.local_bad_density_congruence(Integer(3), Integer(27), None, None) 8/27 
 - local_density(p, m)[source]¶
- Return the local density. - Note - This screens for imprimitive forms, and puts the quadratic form in local normal form, which is a requirement of the routines performing the computations! - INPUT: - p– a prime number > 0
- m– integer
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) # NOTE: This is already in local normal form for *all* primes p! sage: Q.local_density(p=2, m=1) 1 sage: Q.local_density(p=3, m=1) 8/9 sage: Q.local_density(p=5, m=1) 24/25 sage: Q.local_density(p=7, m=1) 48/49 sage: Q.local_density(p=11, m=1) 120/121 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) # NOTE: This is already in local normal form for *all* primes p! >>> Q.local_density(p=Integer(2), m=Integer(1)) 1 >>> Q.local_density(p=Integer(3), m=Integer(1)) 8/9 >>> Q.local_density(p=Integer(5), m=Integer(1)) 24/25 >>> Q.local_density(p=Integer(7), m=Integer(1)) 48/49 >>> Q.local_density(p=Integer(11), m=Integer(1)) 120/121 
 - local_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the local density of \(Q\) representing \(m\) at \(p\), allowing certain congruence conditions mod \(p\). - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_density_congruence(p=2, m=1, Zvec=None, NZvec=None) 1 sage: Q.local_density_congruence(p=3, m=1, Zvec=None, NZvec=None) 8/9 sage: Q.local_density_congruence(p=5, m=1, Zvec=None, NZvec=None) 24/25 sage: Q.local_density_congruence(p=7, m=1, Zvec=None, NZvec=None) 48/49 sage: Q.local_density_congruence(p=11, m=1, Zvec=None, NZvec=None) 120/121 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_density_congruence(p=Integer(2), m=Integer(1), Zvec=None, NZvec=None) 1 >>> Q.local_density_congruence(p=Integer(3), m=Integer(1), Zvec=None, NZvec=None) 8/9 >>> Q.local_density_congruence(p=Integer(5), m=Integer(1), Zvec=None, NZvec=None) 24/25 >>> Q.local_density_congruence(p=Integer(7), m=Integer(1), Zvec=None, NZvec=None) 48/49 >>> Q.local_density_congruence(p=Integer(11), m=Integer(1), Zvec=None, NZvec=None) 120/121 - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_density_congruence(2, 1, None, None) 1 sage: Q.local_density_congruence(2, 2, None, None) 1 sage: Q.local_density_congruence(2, 4, None, None) 3/2 sage: Q.local_density_congruence(3, 1, None, None) 2/3 sage: Q.local_density_congruence(3, 6, None, None) 4/3 sage: Q.local_density_congruence(3, 9, None, None) 14/9 sage: Q.local_density_congruence(3, 27, None, None) 2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_density_congruence(Integer(2), Integer(1), None, None) 1 >>> Q.local_density_congruence(Integer(2), Integer(2), None, None) 1 >>> Q.local_density_congruence(Integer(2), Integer(4), None, None) 3/2 >>> Q.local_density_congruence(Integer(3), Integer(1), None, None) 2/3 >>> Q.local_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_density_congruence(Integer(3), Integer(9), None, None) 14/9 >>> Q.local_density_congruence(Integer(3), Integer(27), None, None) 2 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,3,9,9]) sage: Q.local_density_congruence(3, 1, None, None) 2 sage: Q.local_density_congruence(3, 3, None, None) 4/3 sage: Q.local_density_congruence(3, 6, None, None) 4/3 sage: Q.local_density_congruence(3, 9, None, None) 2/9 sage: Q.local_density_congruence(3, 18, None, None) 4/9 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(9),Integer(9)]) >>> Q.local_density_congruence(Integer(3), Integer(1), None, None) 2 >>> Q.local_density_congruence(Integer(3), Integer(3), None, None) 4/3 >>> Q.local_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_density_congruence(Integer(3), Integer(9), None, None) 2/9 >>> Q.local_density_congruence(Integer(3), Integer(18), None, None) 4/9 
 - local_genus_symbol(p)[source]¶
- Return the Conway-Sloane genus symbol of 2 times a quadratic form defined over \(\ZZ\) at a prime number \(p\). - This is defined (in the class - Genus_Symbol_p_adic_ring) to be a list of tuples (one for each Jordan component \(p^m\cdot A\) at \(p\), where \(A\) is a unimodular symmetric matrix with coefficients the \(p\)-adic integers) of the following form:- If \(p>2\), then return triples of the form [\(m\), \(n\), \(d\)] where - \(m\) = valuation of the component 
- \(n\) = rank of \(A\) 
- \(d\) = det(\(A\)) in {1, \(u\)} for normalized quadratic non-residue \(u\). 
 
- If \(p=2\), then return quintuples of the form [\(m\), \(n\), \(s\), \(d\), \(o\)] where - \(m\) = valuation of the component 
- \(n\) = rank of \(A\) 
- \(d\) = det(\(A\)) in {1, 3, 5, 7} 
- \(s\) = 0 (or 1) if \(A\) is even (or odd) 
- \(o\) = oddity of \(A\) (= 0 if \(s\) = 0) in \(\ZZ/8\ZZ\) = the trace of the diagonalization of \(A\) 
 
 - Note - The Conway-Sloane convention for describing the prime \(p = -1\) is not supported here, and neither is the convention for including the ‘prime’ Infinity. See note on p370 of Conway-Sloane (3rd ed) [CS1999] for a discussion of this convention. - INPUT: - p– a prime number > 0
 - OUTPUT: - a Conway-Sloane genus symbol at \(p\), which is an instance of the class - Genus_Symbol_p_adic_ring.- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3,4]) sage: Q.local_genus_symbol(2) Genus symbol at 2: [2^-2 4^1 8^1]_6 sage: Q.local_genus_symbol(3) Genus symbol at 3: 1^3 3^-1 sage: Q.local_genus_symbol(5) Genus symbol at 5: 1^4 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3),Integer(4)]) >>> Q.local_genus_symbol(Integer(2)) Genus symbol at 2: [2^-2 4^1 8^1]_6 >>> Q.local_genus_symbol(Integer(3)) Genus symbol at 3: 1^3 3^-1 >>> Q.local_genus_symbol(Integer(5)) Genus symbol at 5: 1^4 
 - local_good_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the Good-type local density of \(Q\) representing \(m\) at \(p\). (Front end routine for parity specific routines for \(p\).) - Todo - Add documentation about the additional congruence conditions - Zvecand- NZvec.- INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_good_density_congruence(2, 1, None, None) 1 sage: Q.local_good_density_congruence(3, 1, None, None) 2/3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_good_density_congruence(Integer(2), Integer(1), None, None) 1 >>> Q.local_good_density_congruence(Integer(3), Integer(1), None, None) 2/3 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_good_density_congruence(2, 1, None, None) 1 sage: Q.local_good_density_congruence(3, 1, None, None) 8/9 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_good_density_congruence(Integer(2), Integer(1), None, None) 1 >>> Q.local_good_density_congruence(Integer(3), Integer(1), None, None) 8/9 
 - local_good_density_congruence_even(m, Zvec, NZvec)[source]¶
- Find the Good-type local density of \(Q\) representing \(m\) at \(p=2\). (Assuming \(Q\) is given in local diagonal form.) - The additional congruence condition arguments - Zvecand- NZveccan be either a list of indices or None.- Zvec=[]is equivalent to- Zvec=Nonewhich both impose no additional conditions, but- NZvec=[]returns no solutions always while- NZvec=Noneimposes no additional condition.- Warning - Here the indices passed in - Zvecand- NZvecrepresent indices of the solution vector \(x\) of \(Q(x) = m\) (mod \(p^k\)), and not the Jordan components of \(Q\). They therefore are required (and assumed) to include either all or none of the indices of a given Jordan component of \(Q\). This is only important when \(p=2\) since otherwise all Jordan blocks are \(1 \times 1\), and so there the indices and Jordan blocks coincide.- Todo - Add type checking for - Zvecand- NZvec, and that \(Q\) is in local normal form.- INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and 2-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_good_density_congruence_even(1, None, None) 1 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_good_density_congruence_even(Integer(1), None, None) 1 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_good_density_congruence_even(1, None, None) 1 sage: Q.local_good_density_congruence_even(2, None, None) 3/2 sage: Q.local_good_density_congruence_even(3, None, None) 1 sage: Q.local_good_density_congruence_even(4, None, None) 1/2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_good_density_congruence_even(Integer(1), None, None) 1 >>> Q.local_good_density_congruence_even(Integer(2), None, None) 3/2 >>> Q.local_good_density_congruence_even(Integer(3), None, None) 1 >>> Q.local_good_density_congruence_even(Integer(4), None, None) 1/2 - sage: Q = QuadraticForm(ZZ, 4, range(10)) sage: Q[0,0] = 5 sage: Q[1,1] = 10 sage: Q[2,2] = 15 sage: Q[3,3] = 20 sage: Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 5 1 2 3 ] [ * 10 5 6 ] [ * * 15 8 ] [ * * * 20 ] sage: Q.theta_series(20) # needs sage.libs.pari 1 + 2*q^5 + 2*q^10 + 2*q^14 + 2*q^15 + 2*q^16 + 2*q^18 + O(q^20) sage: Q_local = Q.local_normal_form(2) # needs sage.libs.pari sage.rings.padics sage: Q_local.local_good_density_congruence_even(1, None, None) # needs sage.libs.pari sage.rings.padics 3/4 sage: Q_local.local_good_density_congruence_even(2, None, None) # needs sage.libs.pari sage.rings.padics 9/8 sage: Q_local.local_good_density_congruence_even(5, None, None) # needs sage.libs.pari sage.rings.padics 3/4 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))) >>> Q[Integer(0),Integer(0)] = Integer(5) >>> Q[Integer(1),Integer(1)] = Integer(10) >>> Q[Integer(2),Integer(2)] = Integer(15) >>> Q[Integer(3),Integer(3)] = Integer(20) >>> Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 5 1 2 3 ] [ * 10 5 6 ] [ * * 15 8 ] [ * * * 20 ] >>> Q.theta_series(Integer(20)) # needs sage.libs.pari 1 + 2*q^5 + 2*q^10 + 2*q^14 + 2*q^15 + 2*q^16 + 2*q^18 + O(q^20) >>> Q_local = Q.local_normal_form(Integer(2)) # needs sage.libs.pari sage.rings.padics >>> Q_local.local_good_density_congruence_even(Integer(1), None, None) # needs sage.libs.pari sage.rings.padics 3/4 >>> Q_local.local_good_density_congruence_even(Integer(2), None, None) # needs sage.libs.pari sage.rings.padics 9/8 >>> Q_local.local_good_density_congruence_even(Integer(5), None, None) # needs sage.libs.pari sage.rings.padics 3/4 
 - local_good_density_congruence_odd(p, m, Zvec, NZvec)[source]¶
- Find the Good-type local density of \(Q\) representing \(m\) at \(p\). (Assuming that \(p > 2\) and \(Q\) is given in local diagonal form.) - The additional congruence condition arguments - Zvecand- NZveccan be either a list of indices or None.- Zvec=[]is equivalent to- Zvec=None, which both impose no additional conditions, but- NZvec=[]returns no solutions always while- NZvec=Noneimposes no additional condition.- Todo - Add type checking for - Zvec,- NZvec, and that \(Q\) is in local normal form.- INPUT: - self– quadratic form \(Q\), assumed to be diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_good_density_congruence_odd(3, 1, None, None) 2/3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_good_density_congruence_odd(Integer(3), Integer(1), None, None) 2/3 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_good_density_congruence_odd(3, 1, None, None) 8/9 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_good_density_congruence_odd(Integer(3), Integer(1), None, None) 8/9 
 - local_normal_form(p)[source]¶
- Return a locally integrally equivalent quadratic form over the \(p\)-adic integers \(\ZZ_p\) which gives the Jordan decomposition. - The Jordan components are written as sums of blocks of size \(\leq 2\) and are arranged by increasing scale, and then by increasing norm. This is equivalent to saying that we put the \(1 \times 1\) blocks before the \(2 \times 2\) blocks in each Jordan component. - INPUT: - p– a positive prime number
 - OUTPUT: a quadratic form over \(\ZZ\) - Warning - Currently this only works for quadratic forms defined over \(\ZZ\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [10,4,1]) sage: Q.local_normal_form(5) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 6 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(10),Integer(4),Integer(1)]) >>> Q.local_normal_form(Integer(5)) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 6 ] - sage: Q.local_normal_form(3) Quadratic form in 2 variables over Integer Ring with coefficients: [ 10 0 ] [ * 15 ] sage: Q.local_normal_form(2) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 6 ] - >>> from sage.all import * >>> Q.local_normal_form(Integer(3)) Quadratic form in 2 variables over Integer Ring with coefficients: [ 10 0 ] [ * 15 ] >>> Q.local_normal_form(Integer(2)) Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 0 ] [ * 6 ] 
 - local_primitive_density(p, m)[source]¶
- Return the local primitive density – should be called by the user. - NOTE: This screens for imprimitive forms, and puts the quadratic form in local normal form, which is a requirement of the routines performing the computations! - INPUT: - p– a prime number > 0
- m– integer
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(10)) sage: Q[0,0] = 5 sage: Q[1,1] = 10 sage: Q[2,2] = 15 sage: Q[3,3] = 20 sage: Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 5 1 2 3 ] [ * 10 5 6 ] [ * * 15 8 ] [ * * * 20 ] sage: Q.theta_series(20) # needs sage.libs.pari 1 + 2*q^5 + 2*q^10 + 2*q^14 + 2*q^15 + 2*q^16 + 2*q^18 + O(q^20) sage: Q.local_normal_form(2) Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 0 0 ] [ * 0 0 0 ] [ * * 0 1 ] [ * * * 0 ] sage: Q.local_primitive_density(2, 1) 3/4 sage: Q.local_primitive_density(5, 1) 24/25 sage: Q.local_primitive_density(2, 5) 3/4 sage: Q.local_density(2, 5) 3/4 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))) >>> Q[Integer(0),Integer(0)] = Integer(5) >>> Q[Integer(1),Integer(1)] = Integer(10) >>> Q[Integer(2),Integer(2)] = Integer(15) >>> Q[Integer(3),Integer(3)] = Integer(20) >>> Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 5 1 2 3 ] [ * 10 5 6 ] [ * * 15 8 ] [ * * * 20 ] >>> Q.theta_series(Integer(20)) # needs sage.libs.pari 1 + 2*q^5 + 2*q^10 + 2*q^14 + 2*q^15 + 2*q^16 + 2*q^18 + O(q^20) >>> Q.local_normal_form(Integer(2)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 0 0 ] [ * 0 0 0 ] [ * * 0 1 ] [ * * * 0 ] >>> Q.local_primitive_density(Integer(2), Integer(1)) 3/4 >>> Q.local_primitive_density(Integer(5), Integer(1)) 24/25 >>> Q.local_primitive_density(Integer(2), Integer(5)) 3/4 >>> Q.local_density(Integer(2), Integer(5)) 3/4 
 - local_primitive_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the primitive local density of \(Q\) representing \(m\) at \(p\), allowing certain congruence conditions mod \(p\). - Note - The following routine is not used internally, but is included for consistency. - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_primitive_density_congruence(p=2, m=1, Zvec=None, NZvec=None) 1 sage: Q.local_primitive_density_congruence(p=3, m=1, Zvec=None, NZvec=None) 8/9 sage: Q.local_primitive_density_congruence(p=5, m=1, Zvec=None, NZvec=None) 24/25 sage: Q.local_primitive_density_congruence(p=7, m=1, Zvec=None, NZvec=None) 48/49 sage: Q.local_primitive_density_congruence(p=11, m=1, Zvec=None, NZvec=None) 120/121 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_primitive_density_congruence(p=Integer(2), m=Integer(1), Zvec=None, NZvec=None) 1 >>> Q.local_primitive_density_congruence(p=Integer(3), m=Integer(1), Zvec=None, NZvec=None) 8/9 >>> Q.local_primitive_density_congruence(p=Integer(5), m=Integer(1), Zvec=None, NZvec=None) 24/25 >>> Q.local_primitive_density_congruence(p=Integer(7), m=Integer(1), Zvec=None, NZvec=None) 48/49 >>> Q.local_primitive_density_congruence(p=Integer(11), m=Integer(1), Zvec=None, NZvec=None) 120/121 - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_primitive_density_congruence(2, 1, None, None) 1 sage: Q.local_primitive_density_congruence(2, 2, None, None) 1 sage: Q.local_primitive_density_congruence(2, 4, None, None) 1 sage: Q.local_primitive_density_congruence(3, 1, None, None) 2/3 sage: Q.local_primitive_density_congruence(3, 6, None, None) 4/3 sage: Q.local_primitive_density_congruence(3, 9, None, None) 4/3 sage: Q.local_primitive_density_congruence(3, 27, None, None) 4/3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_primitive_density_congruence(Integer(2), Integer(1), None, None) 1 >>> Q.local_primitive_density_congruence(Integer(2), Integer(2), None, None) 1 >>> Q.local_primitive_density_congruence(Integer(2), Integer(4), None, None) 1 >>> Q.local_primitive_density_congruence(Integer(3), Integer(1), None, None) 2/3 >>> Q.local_primitive_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_primitive_density_congruence(Integer(3), Integer(9), None, None) 4/3 >>> Q.local_primitive_density_congruence(Integer(3), Integer(27), None, None) 4/3 - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,3,9,9]) sage: Q.local_primitive_density_congruence(3, 1, None, None) 2 sage: Q.local_primitive_density_congruence(3, 3, None, None) 4/3 sage: Q.local_primitive_density_congruence(3, 6, None, None) 4/3 sage: Q.local_primitive_density_congruence(3, 9, None, None) 4/27 sage: Q.local_primitive_density_congruence(3, 18, None, None) 4/9 sage: Q.local_primitive_density_congruence(3, 27, None, None) 8/27 sage: Q.local_primitive_density_congruence(3, 81, None, None) 8/27 sage: Q.local_primitive_density_congruence(3, 243, None, None) 8/27 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(9),Integer(9)]) >>> Q.local_primitive_density_congruence(Integer(3), Integer(1), None, None) 2 >>> Q.local_primitive_density_congruence(Integer(3), Integer(3), None, None) 4/3 >>> Q.local_primitive_density_congruence(Integer(3), Integer(6), None, None) 4/3 >>> Q.local_primitive_density_congruence(Integer(3), Integer(9), None, None) 4/27 >>> Q.local_primitive_density_congruence(Integer(3), Integer(18), None, None) 4/9 >>> Q.local_primitive_density_congruence(Integer(3), Integer(27), None, None) 8/27 >>> Q.local_primitive_density_congruence(Integer(3), Integer(81), None, None) 8/27 >>> Q.local_primitive_density_congruence(Integer(3), Integer(243), None, None) 8/27 
 - local_representation_conditions(recompute_flag=False, silent_flag=False)[source]¶
- Warning - This only works correctly for forms in >=3 variables, which are locally universal at almost all primes! - This class finds the local conditions for a number to be integrally represented by an integer-valued quadratic form. These conditions are stored in - self.__local_representability_conditionsand consist of a list of 9 element vectors, with one for each prime with a local obstruction (though only the first 5 are meaningful unless \(p=2\)). The first element is always the prime \(p\) where the local obstruction occurs, and the next 8 (or 4) entries represent square-classes in the \(p\)-adic integers \(\ZZ_p\), and are labeled by the \(\QQ_p\) square-classes \(t\cdot (\QQ_p)^2\) with \(t\) given as follows:- for \(p > 2\), - [ * 1 u p u p * * * * ],
- for \(p = 2\), - [ * 1 3 5 7 2 6 10 14 ].
 - The integer appearing in each place tells us how \(p\)-divisible a number needs to be in that square-class in order to be locally represented by \(Q\). A negative number indicates that the entire \(\QQ_p\) square-class is not represented, while a positive number \(x\) indicates that \(t\cdot p^{(2\cdot x)} (\ZZ_p)^2\) is locally represented but \(t\cdot p^{(2\cdot (x-1))}\) \((\ZZ_p)^2\) is not. - As an example, the vector - [2 3 0 0 0 0 2 0 infinity]tells us that all positive integers are locally represented at \(p=2\) except those of the forms:- \(2^6\cdot u\cdot r^2\) with \(u = 1\) (mod 8) 
- \(2^5\cdot u\cdot r^2\) with \(u = 3\) (mod 8) 
- \(2\cdot u\cdot r^2\) with \(u = 7\) (mod 8) 
 - At the real numbers, the vector which looks like - [infinity, 0, infinity, None, None, None, None, None, None]means that \(Q\) is negative definite (i.e., the 0 tells us all positive reals are represented). The real vector always appears, and is listed before the other ones.- OUTPUT: - A list of 9-element vectors describing the representation obstructions at primes dividing the level. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, []) sage: Q.local_representation_conditions() This 0-dimensional form only represents zero. sage: Q = DiagonalQuadraticForm(ZZ, [5]) sage: Q.local_representation_conditions() This 1-dimensional form only represents square multiples of 5. sage: Q1 = DiagonalQuadraticForm(ZZ, [1,1]) sage: Q1.local_representation_conditions() This 2-dimensional form represents the p-adic integers of even valuation for all primes p except [2]. For these and the reals, we have: Reals: [0, +Infinity] p = 2: [0, +Infinity, 0, +Infinity, 0, +Infinity, 0, +Infinity] sage: Q1 = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [2]. For these and the reals, we have: Reals: [0, +Infinity] p = 2: [0, 0, 0, +Infinity, 0, 0, 0, 0] sage: Q1 = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except []. For these and the reals, we have: Reals: [0, +Infinity] sage: Q1 = DiagonalQuadraticForm(ZZ, [1,3,3,3]) sage: Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [3]. For these and the reals, we have: Reals: [0, +Infinity] p = 3: [0, 1, 0, 0] sage: Q2 = DiagonalQuadraticForm(ZZ, [2,3,3,3]) sage: Q2.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [3]. For these and the reals, we have: Reals: [0, +Infinity] p = 3: [1, 0, 0, 0] sage: Q3 = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q3.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except []. For these and the reals, we have: Reals: [0, +Infinity] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, []) >>> Q.local_representation_conditions() This 0-dimensional form only represents zero. >>> Q = DiagonalQuadraticForm(ZZ, [Integer(5)]) >>> Q.local_representation_conditions() This 1-dimensional form only represents square multiples of 5. >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]) >>> Q1.local_representation_conditions() This 2-dimensional form represents the p-adic integers of even valuation for all primes p except [2]. For these and the reals, we have: Reals: [0, +Infinity] p = 2: [0, +Infinity, 0, +Infinity, 0, +Infinity, 0, +Infinity] >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [2]. For these and the reals, we have: Reals: [0, +Infinity] p = 2: [0, 0, 0, +Infinity, 0, 0, 0, 0] >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except []. For these and the reals, we have: Reals: [0, +Infinity] >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(3),Integer(3)]) >>> Q1.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [3]. For these and the reals, we have: Reals: [0, +Infinity] p = 3: [0, 1, 0, 0] >>> Q2 = DiagonalQuadraticForm(ZZ, [Integer(2),Integer(3),Integer(3),Integer(3)]) >>> Q2.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except [3]. For these and the reals, we have: Reals: [0, +Infinity] p = 3: [1, 0, 0, 0] >>> Q3 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q3.local_representation_conditions() This form represents the p-adic integers Z_p for all primes p except []. For these and the reals, we have: Reals: [0, +Infinity] 
 - local_zero_density_congruence(p, m, Zvec=None, NZvec=None)[source]¶
- Find the Zero-type local density of \(Q\) representing \(m\) at \(p\), allowing certain congruence conditions mod \(p\). - INPUT: - self– quadratic form \(Q\), assumed to be block diagonal and \(p\)-integral
- p– a prime number
- m– integer
- Zvec,- NZvec– non-repeating lists of integers in- range(self.dim())or- None
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,3]) sage: Q.local_zero_density_congruence(2, 2, None, None) 0 sage: Q.local_zero_density_congruence(2, 4, None, None) 1/2 sage: Q.local_zero_density_congruence(3, 6, None, None) 0 sage: Q.local_zero_density_congruence(3, 9, None, None) 2/9 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),Integer(3)]) >>> Q.local_zero_density_congruence(Integer(2), Integer(2), None, None) 0 >>> Q.local_zero_density_congruence(Integer(2), Integer(4), None, None) 1/2 >>> Q.local_zero_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_zero_density_congruence(Integer(3), Integer(9), None, None) 2/9 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.local_zero_density_congruence(2, 2, None, None) 0 sage: Q.local_zero_density_congruence(2, 4, None, None) 1/4 sage: Q.local_zero_density_congruence(3, 6, None, None) 0 sage: Q.local_zero_density_congruence(3, 9, None, None) 8/81 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.local_zero_density_congruence(Integer(2), Integer(2), None, None) 0 >>> Q.local_zero_density_congruence(Integer(2), Integer(4), None, None) 1/4 >>> Q.local_zero_density_congruence(Integer(3), Integer(6), None, None) 0 >>> Q.local_zero_density_congruence(Integer(3), Integer(9), None, None) 8/81 
 - mass__by_Siegel_densities(odd_algorithm='Pall', even_algorithm='Watson')[source]¶
- Return the mass of transformations (det 1 and -1). - Warning - This is broken right now… - INPUT: - odd_algorithm– algorithm to be used when \(p>2\);- 'Pall'(only one choice for now)
- even_algorithm– algorithm to be used when \(p=2\); either- 'Kitaoka'or- 'Watson'(the default)
 - REFERENCES: - Nipp’s Book “Tables of Quaternary Quadratic Forms”. 
- Papers of Pall (only for \(p>2\)) and Watson (for \(p=2\) – tricky!). 
- Siegel, Milnor-Hussemoller, Conway-Sloane Paper IV, Kitoaka (all of which have problems…) 
 - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: m = Q.mass__by_Siegel_densities(); m # needs sage.symbolic 1/384 sage: m - (2^Q.dim() * factorial(Q.dim()))^(-1) # needs sage.symbolic 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> m = Q.mass__by_Siegel_densities(); m # needs sage.symbolic 1/384 >>> m - (Integer(2)**Q.dim() * factorial(Q.dim()))**(-Integer(1)) # needs sage.symbolic 0 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: m = Q.mass__by_Siegel_densities(); m # needs sage.symbolic 1/48 sage: m - (2^Q.dim() * factorial(Q.dim()))^(-1) # needs sage.symbolic 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> m = Q.mass__by_Siegel_densities(); m # needs sage.symbolic 1/48 >>> m - (Integer(2)**Q.dim() * factorial(Q.dim()))**(-Integer(1)) # needs sage.symbolic 0 
 - mass_at_two_by_counting_mod_power(k)[source]¶
- Compute the local mass at \(p=2\) assuming that it’s stable (mod \(2^k\)). - Note - This is way too slow to be useful, even when \(k=1\). - Todo - Remove this routine, or try to compile it! - INPUT: - k– integer \(\geq 1\)
 - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.mass_at_two_by_counting_mod_power(1) 4 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.mass_at_two_by_counting_mod_power(Integer(1)) 4 
 - matrix()[source]¶
- Return the Hessian matrix \(A\) for which \(Q(X) = (1/2) X^t\cdot A\cdot X\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, range(6)) sage: Q.matrix() [ 0 1 2] [ 1 6 4] [ 2 4 10] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), range(Integer(6))) >>> Q.matrix() [ 0 1 2] [ 1 6 4] [ 2 4 10] 
 - minkowski_reduction()[source]¶
- Find a Minkowski-reduced form equivalent to the given one. - This means that \[Q(v_k) \leq Q(s_1\cdot v_1 + ... + s_n\cdot v_n)\]- for all \(s_i\) where \(\gcd(s_k, ... s_n) = 1\). - Note - When \(Q\) has dim \(\leq 4\) we can take all \(s_i\) in \(\{1, 0, -1\}\). - REFERENCES: - Schulze-Pillot’s paper on “An algorithm for computing genera of ternary and quaternary quadratic forms”, p138. 
- Donaldson’s 1979 paper “Minkowski Reduction of Integral Matrices”, p203. 
 - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, [30, 17, 11, 12, 29, 25, 62, 64, 25, 110]) sage: Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 12 ] [ * 29 25 62 ] [ * * 64 25 ] [ * * * 110 ] sage: Q.minkowski_reduction() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 -5 ] [ * 29 25 4 ] [ * * 64 0 ] [ * * * 77 ] , [ 1 0 0 0] [ 0 1 0 -1] [ 0 0 1 0] [ 0 0 0 1] ) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), [Integer(30), Integer(17), Integer(11), Integer(12), Integer(29), Integer(25), Integer(62), Integer(64), Integer(25), Integer(110)]) >>> Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 12 ] [ * 29 25 62 ] [ * * 64 25 ] [ * * * 110 ] >>> Q.minkowski_reduction() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 -5 ] [ * 29 25 4 ] [ * * 64 0 ] [ * * * 77 ] , <BLANKLINE> [ 1 0 0 0] [ 0 1 0 -1] [ 0 0 1 0] [ 0 0 0 1] ) - sage: Q = QuadraticForm(ZZ,4,[1, -2, 0, 0, 2, 0, 0, 2, 0, 2]) sage: Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 -2 0 0 ] [ * 2 0 0 ] [ * * 2 0 ] [ * * * 2 ] sage: Q.minkowski_reduction() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 0 0 ] [ * * 2 0 ] [ * * * 2 ] , [1 1 0 0] [0 1 0 0] [0 0 1 0] [0 0 0 1] ) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ,Integer(4),[Integer(1), -Integer(2), Integer(0), Integer(0), Integer(2), Integer(0), Integer(0), Integer(2), Integer(0), Integer(2)]) >>> Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 -2 0 0 ] [ * 2 0 0 ] [ * * 2 0 ] [ * * * 2 ] >>> Q.minkowski_reduction() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 1 0 0 ] [ * * 2 0 ] [ * * * 2 ] , <BLANKLINE> [1 1 0 0] [0 1 0 0] [0 0 1 0] [0 0 0 1] ) 
 - minkowski_reduction_for_4vars__SP()[source]¶
- Find a Minkowski-reduced form equivalent to the given one. This means that \[Q(v_k) \leq Q(s_1\cdot v_1 + ... + s_n\cdot v_n)\]- for all \(s_i\) where GCD(\(s_k, ... s_n\)) = 1. - Note - When \(Q\) has dim \(\leq 4\), we can take all \(s_i\) in \(\{1, 0, -1\}\). - REFERENCES: - Schulze-Pillot’s paper on “An algorithm for computing genera of ternary and quaternary quadratic forms”, p138. 
- Donaldson’s 1979 paper “Minkowski Reduction of Integral Matrices”, p203. 
 - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, [30,17,11,12,29,25,62,64,25,110]); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 12 ] [ * 29 25 62 ] [ * * 64 25 ] [ * * * 110 ] sage: Q.minkowski_reduction_for_4vars__SP() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 29 -17 25 4 ] [ * 30 -11 5 ] [ * * 64 0 ] [ * * * 77 ] , [ 0 1 0 0] [ 1 0 0 -1] [ 0 0 1 0] [ 0 0 0 1] ) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), [Integer(30),Integer(17),Integer(11),Integer(12),Integer(29),Integer(25),Integer(62),Integer(64),Integer(25),Integer(110)]); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 30 17 11 12 ] [ * 29 25 62 ] [ * * 64 25 ] [ * * * 110 ] >>> Q.minkowski_reduction_for_4vars__SP() ( Quadratic form in 4 variables over Integer Ring with coefficients: [ 29 -17 25 4 ] [ * 30 -11 5 ] [ * * 64 0 ] [ * * * 77 ] , <BLANKLINE> [ 0 1 0 0] [ 1 0 0 -1] [ 0 0 1 0] [ 0 0 0 1] ) 
 - multiply_variable(c, i, in_place=False)[source]¶
- Replace the variables \(x_i\) by \(c\cdot x_i\) in the quadratic form (replacing the original form if the - in_placeflag is True).- Here \(c\) must be an element of the base ring defining the quadratic form. - INPUT: - c– an element of- self.base_ring()
- i– integer \(\geq 0\)
 - OUTPUT: a - QuadraticForm(by default, otherwise none)- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,9,5,7]) sage: Q.multiply_variable(5, 0) Quadratic form in 4 variables over Integer Ring with coefficients: [ 25 0 0 0 ] [ * 9 0 0 ] [ * * 5 0 ] [ * * * 7 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(9),Integer(5),Integer(7)]) >>> Q.multiply_variable(Integer(5), Integer(0)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 25 0 0 0 ] [ * 9 0 0 ] [ * * 5 0 ] [ * * * 7 ] 
 - neighbor_iteration(seeds, p, mass=None, max_classes=None, algorithm=None, max_neighbors=1000, verbose=False)[source]¶
- Return all classes in the \(p\)-neighbor graph of - self.- Starting from the given seeds, this function successively finds \(p\)-neighbors until no new quadratic form (class) is obtained. - INPUT: - seeds– list of quadratic forms in the same genus
- p– a prime number
- mass– (optional) a rational number; the mass of this genus
- max_classes– (default:- 1000) break the computation when- max_classesare found
- algorithm– (optional) one of- 'orbits',- 'random',- 'exhaustion'
- max_random_trys– (default:- 1000) the maximum number of neighbors computed for a single lattice
 - OUTPUT: list of quadratic forms - EXAMPLES: - sage: from sage.quadratic_forms.quadratic_form__neighbors import neighbor_iteration sage: Q = QuadraticForm(ZZ, 3, [1, 0, 0, 2, 1, 3]) sage: Q.det() 46 sage: # needs sage.symbolic sage: mass = Q.conway_mass() sage: g1 = neighbor_iteration([Q], 3, # long time ....: mass=mass, algorithm='random') sage: g2 = neighbor_iteration([Q], 3, algorithm='exhaustion') # long time sage: g3 = neighbor_iteration([Q], 3, algorithm='orbits') # needs sage.libs.gap sage: mass == sum(1/q.number_of_automorphisms() for q in g1) # long time True sage: mass == sum(1/q.number_of_automorphisms() for q in g2) # long time True sage: mass == sum(1/q.number_of_automorphisms() for q in g3) # needs sage.libs.gap True - >>> from sage.all import * >>> from sage.quadratic_forms.quadratic_form__neighbors import neighbor_iteration >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), Integer(0), Integer(2), Integer(1), Integer(3)]) >>> Q.det() 46 >>> # needs sage.symbolic >>> mass = Q.conway_mass() >>> g1 = neighbor_iteration([Q], Integer(3), # long time ... mass=mass, algorithm='random') >>> g2 = neighbor_iteration([Q], Integer(3), algorithm='exhaustion') # long time >>> g3 = neighbor_iteration([Q], Integer(3), algorithm='orbits') # needs sage.libs.gap >>> mass == sum(Integer(1)/q.number_of_automorphisms() for q in g1) # long time True >>> mass == sum(Integer(1)/q.number_of_automorphisms() for q in g2) # long time True >>> mass == sum(Integer(1)/q.number_of_automorphisms() for q in g3) # needs sage.libs.gap True 
 - number_of_automorphisms()[source]¶
- Return the number of automorphisms (of det \(1\) and \(-1\)) of the quadratic form. - OUTPUT: integer \(\geq 2\) - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [1, 0, 0, 1, 0, 1], unsafe_initialization=True) sage: Q.number_of_automorphisms() 48 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), Integer(0), Integer(1), Integer(0), Integer(1)], unsafe_initialization=True) >>> Q.number_of_automorphisms() 48 - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.number_of_automorphisms() 384 sage: 2^4 * factorial(4) 384 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.number_of_automorphisms() 384 >>> Integer(2)**Integer(4) * factorial(Integer(4)) 384 
 - omega()[source]¶
- Return the content of the adjoint of the primitive associated quadratic form. - Ref: See Dickson’s “Studies in Number Theory”. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,37]) sage: Q.omega() 4 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(37)]) >>> Q.omega() 4 
 - orbits_lines_mod_p(p)[source]¶
- Let \((L, q)\) be a lattice. This returns representatives of the orbits of lines in \(L/pL\) under the orthogonal group of \(q\). - INPUT: - p– a prime number
 - OUTPUT: list of vectors over - GF(p)- EXAMPLES: - sage: from sage.quadratic_forms.quadratic_form__neighbors import orbits_lines_mod_p sage: Q = QuadraticForm(ZZ, 3, [1, 0, 0, 2, 1, 3]) sage: Q.orbits_lines_mod_p(2) # needs sage.libs.gap sage.libs.pari [(0, 0, 1), (0, 1, 0), (0, 1, 1), (1, 0, 0), (1, 0, 1), (1, 1, 0), (1, 1, 1)] - >>> from sage.all import * >>> from sage.quadratic_forms.quadratic_form__neighbors import orbits_lines_mod_p >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(0), Integer(0), Integer(2), Integer(1), Integer(3)]) >>> Q.orbits_lines_mod_p(Integer(2)) # needs sage.libs.gap sage.libs.pari [(0, 0, 1), (0, 1, 0), (0, 1, 1), (1, 0, 0), (1, 0, 1), (1, 1, 0), (1, 1, 1)] 
 - parity(allow_rescaling_flag=True)[source]¶
- Return the parity (“even” or “odd”) of an integer-valued quadratic form over \(\ZZ\), defined up to similitude/rescaling of the form so that its Jordan component of smallest scale is unimodular. After this rescaling, we say a form is even if it only represents even numbers, and odd if it represents some odd number. - If the - allow_rescaling_flagis set to False, then we require that the quadratic form have a Gram matrix with coefficients in \(\ZZ\), and look at the unimodular Jordan block to determine its parity. This returns an error if the form is not integer-matrix, meaning that it has Jordan components at \(p=2\) which do not have an integer scale.- We determine the parity by looking for a \(1 \times 1\) block in the 0-th Jordan component, after a possible rescaling. - INPUT: - self– a quadratic form with base ring \(\ZZ\), which we may require to have integer Gram matrix
 - OUTPUT: one of the strings: - 'even'or- 'odd'- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 3, [4, -2, 0, 2, 3, 2]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 3 ] [ * * 2 ] sage: Q.parity() 'even' - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(4), -Integer(2), Integer(0), Integer(2), Integer(3), Integer(2)]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 3 ] [ * * 2 ] >>> Q.parity() 'even' - sage: Q = QuadraticForm(ZZ, 3, [4, -2, 0, 2, 3, 1]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 3 ] [ * * 1 ] sage: Q.parity() 'even' - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(4), -Integer(2), Integer(0), Integer(2), Integer(3), Integer(1)]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 3 ] [ * * 1 ] >>> Q.parity() 'even' - sage: Q = QuadraticForm(ZZ, 3, [4, -2, 0, 2, 2, 2]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 2 ] [ * * 2 ] sage: Q.parity() 'even' - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(4), -Integer(2), Integer(0), Integer(2), Integer(2), Integer(2)]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 2 ] [ * * 2 ] >>> Q.parity() 'even' - sage: Q = QuadraticForm(ZZ, 3, [4, -2, 0, 2, 2, 1]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 2 ] [ * * 1 ] sage: Q.parity() 'odd' - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(3), [Integer(4), -Integer(2), Integer(0), Integer(2), Integer(2), Integer(1)]); Q Quadratic form in 3 variables over Integer Ring with coefficients: [ 4 -2 0 ] [ * 2 2 ] [ * * 1 ] >>> Q.parity() 'odd' 
 - polynomial(names='x')[source]¶
- Return the quadratic form as a polynomial in \(n\) variables. - INPUT: - self– a quadratic form over a commutative ring
- names– specification of the names of the variables; see- PolynomialRing()
 - OUTPUT: the polynomial form of the quadratic form - EXAMPLES: - sage: Q = DiagonalQuadraticForm(QQ,[1, 3, 5, 7]) sage: P = Q.polynomial(); P x0^2 + 3*x1^2 + 5*x2^2 + 7*x3^2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(QQ,[Integer(1), Integer(3), Integer(5), Integer(7)]) >>> P = Q.polynomial(); P x0^2 + 3*x1^2 + 5*x2^2 + 7*x3^2 - sage: # needs sage.rings.number_field sage: x = polygen(ZZ, 'x') sage: F.<a> = NumberField(x^2 - 5) sage: Z = F.ring_of_integers() sage: Q = QuadraticForm(Z, 3, [2*a, 3*a, 0, 1 - a, 0, 2*a + 4]) sage: P = Q.polynomial(names='y'); P 2*a*y0^2 + 3*a*y0*y1 + (-a + 1)*y1^2 + (2*a + 4)*y2^2 sage: Q = QuadraticForm(F, 4, ....: [a, 3*a, 0, 1 - a, a - 3, 0, 2*a + 4, 4 + a, 0, 1]) sage: Q.polynomial(names='z') a*z0^2 + (3*a)*z0*z1 + (a - 3)*z1^2 + (a + 4)*z2^2 + (-a + 1)*z0*z3 + (2*a + 4)*z1*z3 + z3^2 sage: B.<i,j,k> = QuaternionAlgebra(F,-1,-1) sage: Q = QuadraticForm(B, 3, [2*a, 3*a, i, 1 - a, 0, 2*a + 4]) sage: Q.polynomial() Traceback (most recent call last): ... ValueError: Can only create polynomial rings over commutative rings - >>> from sage.all import * >>> # needs sage.rings.number_field >>> x = polygen(ZZ, 'x') >>> F = NumberField(x**Integer(2) - Integer(5), names=('a',)); (a,) = F._first_ngens(1) >>> Z = F.ring_of_integers() >>> Q = QuadraticForm(Z, Integer(3), [Integer(2)*a, Integer(3)*a, Integer(0), Integer(1) - a, Integer(0), Integer(2)*a + Integer(4)]) >>> P = Q.polynomial(names='y'); P 2*a*y0^2 + 3*a*y0*y1 + (-a + 1)*y1^2 + (2*a + 4)*y2^2 >>> Q = QuadraticForm(F, Integer(4), ... [a, Integer(3)*a, Integer(0), Integer(1) - a, a - Integer(3), Integer(0), Integer(2)*a + Integer(4), Integer(4) + a, Integer(0), Integer(1)]) >>> Q.polynomial(names='z') a*z0^2 + (3*a)*z0*z1 + (a - 3)*z1^2 + (a + 4)*z2^2 + (-a + 1)*z0*z3 + (2*a + 4)*z1*z3 + z3^2 >>> B = QuaternionAlgebra(F,-Integer(1),-Integer(1), names=('i', 'j', 'k',)); (i, j, k,) = B._first_ngens(3) >>> Q = QuadraticForm(B, Integer(3), [Integer(2)*a, Integer(3)*a, i, Integer(1) - a, Integer(0), Integer(2)*a + Integer(4)]) >>> Q.polynomial() Traceback (most recent call last): ... ValueError: Can only create polynomial rings over commutative rings 
 - primitive()[source]¶
- Return a primitive version of an integer-valued quadratic form, defined over \(\ZZ\). - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [2,3,4]) sage: Q.primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 3 ] [ * 4 ] sage: Q = QuadraticForm(ZZ, 2, [2,4,8]) sage: Q.primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 2 ] [ * 4 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2),Integer(3),Integer(4)]) >>> Q.primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 3 ] [ * 4 ] >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(2),Integer(4),Integer(8)]) >>> Q.primitive() Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 2 ] [ * 4 ] 
 - rational_diagonal_form(return_matrix=False)[source]¶
- Return a diagonal form equivalent to the given quadratic from over the fraction field of its defining ring. - INPUT: - return_matrix– boolean (default:- False); also return the transformation matrix
 - OUTPUT: either the diagonal quadratic form \(D\) (if - return_matrixis false) or the pair \((D, T)\) (if- return_matrixis true) where- D– the diagonalized form of this quadratic form
- T– transformation matrix. This is such that- T.transpose() * self.matrix() * Tgives- D.matrix()
 - Both \(D\) and \(T\) are defined over the fraction field of the base ring of the given form. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [0,1,-1]) sage: Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 0 1 ] [ * -1 ] sage: Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1/4 0 ] [ * -1 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(0),Integer(1),-Integer(1)]) >>> Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 0 1 ] [ * -1 ] >>> Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1/4 0 ] [ * -1 ] - If we start with a diagonal form, we get back the same form defined over the fraction field: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.rational_diagonal_form() Quadratic form in 4 variables over Rational Field with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 5 0 ] [ * * * 7 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.rational_diagonal_form() Quadratic form in 4 variables over Rational Field with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 5 0 ] [ * * * 7 ] - In the following example, we check the consistency of the transformation matrix: - sage: Q = QuadraticForm(ZZ, 4, range(10)) sage: D, T = Q.rational_diagonal_form(return_matrix=True) sage: D Quadratic form in 4 variables over Rational Field with coefficients: [ -1/16 0 0 0 ] [ * 4 0 0 ] [ * * 13 0 ] [ * * * 563/52 ] sage: T [ 1 0 11 149/26] [ -1/8 1 -2 -10/13] [ 0 0 1 -29/26] [ 0 0 0 1] sage: T.transpose() * Q.matrix() * T [ -1/8 0 0 0] [ 0 8 0 0] [ 0 0 26 0] [ 0 0 0 563/26] sage: D.matrix() [ -1/8 0 0 0] [ 0 8 0 0] [ 0 0 26 0] [ 0 0 0 563/26] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))) >>> D, T = Q.rational_diagonal_form(return_matrix=True) >>> D Quadratic form in 4 variables over Rational Field with coefficients: [ -1/16 0 0 0 ] [ * 4 0 0 ] [ * * 13 0 ] [ * * * 563/52 ] >>> T [ 1 0 11 149/26] [ -1/8 1 -2 -10/13] [ 0 0 1 -29/26] [ 0 0 0 1] >>> T.transpose() * Q.matrix() * T [ -1/8 0 0 0] [ 0 8 0 0] [ 0 0 26 0] [ 0 0 0 563/26] >>> D.matrix() [ -1/8 0 0 0] [ 0 8 0 0] [ 0 0 26 0] [ 0 0 0 563/26] - sage: Q1 = QuadraticForm(ZZ, 4, [1, 1, 0, 0, 1, 0, 0, 1, 0, 18]) sage: Q1 Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 1 0 0 ] [ * 1 0 0 ] [ * * 1 0 ] [ * * * 18 ] sage: Q1.rational_diagonal_form(return_matrix=True) ( Quadratic form in 4 variables over Rational Field with coefficients: [ 1 0 0 0 ] [ * 3/4 0 0 ] [ * * 1 0 ] [ * * * 18 ] , [ 1 -1/2 0 0] [ 0 1 0 0] [ 0 0 1 0] [ 0 0 0 1] ) - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(4), [Integer(1), Integer(1), Integer(0), Integer(0), Integer(1), Integer(0), Integer(0), Integer(1), Integer(0), Integer(18)]) >>> Q1 Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 1 0 0 ] [ * 1 0 0 ] [ * * 1 0 ] [ * * * 18 ] >>> Q1.rational_diagonal_form(return_matrix=True) ( Quadratic form in 4 variables over Rational Field with coefficients: [ 1 0 0 0 ] [ * 3/4 0 0 ] [ * * 1 0 ] [ * * * 18 ] , <BLANKLINE> [ 1 -1/2 0 0] [ 0 1 0 0] [ 0 0 1 0] [ 0 0 0 1] ) - PARI returns a singular transformation matrix for this case: - sage: Q = QuadraticForm(QQ, 2, [1/2, 1, 1/2]) sage: Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1/2 0 ] [ * 0 ] - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(2), [Integer(1)/Integer(2), Integer(1), Integer(1)/Integer(2)]) >>> Q.rational_diagonal_form() Quadratic form in 2 variables over Rational Field with coefficients: [ 1/2 0 ] [ * 0 ] - This example cannot be computed by PARI: - sage: Q = QuadraticForm(RIF, 4, range(10)) sage: Q.__pari__() Traceback (most recent call last): ... TypeError sage: Q.rational_diagonal_form() Quadratic form in 4 variables over Real Interval Field with 53 bits of precision with coefficients: [ 5 0.?e-14 0.?e-13 0.?e-13 ] [ * -0.05000000000000? 0.?e-12 0.?e-12 ] [ * * 13.00000000000? 0.?e-10 ] [ * * * 10.8269230769? ] - >>> from sage.all import * >>> Q = QuadraticForm(RIF, Integer(4), range(Integer(10))) >>> Q.__pari__() Traceback (most recent call last): ... TypeError >>> Q.rational_diagonal_form() Quadratic form in 4 variables over Real Interval Field with 53 bits of precision with coefficients: [ 5 0.?e-14 0.?e-13 0.?e-13 ] [ * -0.05000000000000? 0.?e-12 0.?e-12 ] [ * * 13.00000000000? 0.?e-10 ] [ * * * 10.8269230769? ] 
 - reciprocal()[source]¶
- This gives the reciprocal quadratic form associated to the given form. - This is defined as the multiple of the primitive adjoint with the same content as the given form. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,37]) sage: Q.reciprocal() Quadratic form in 3 variables over Integer Ring with coefficients: [ 37 0 0 ] [ * 37 0 ] [ * * 1 ] sage: Q.reciprocal().reciprocal() Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 1 0 ] [ * * 37 ] sage: Q.reciprocal().reciprocal() == Q True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(37)]) >>> Q.reciprocal() Quadratic form in 3 variables over Integer Ring with coefficients: [ 37 0 0 ] [ * 37 0 ] [ * * 1 ] >>> Q.reciprocal().reciprocal() Quadratic form in 3 variables over Integer Ring with coefficients: [ 1 0 0 ] [ * 1 0 ] [ * * 37 ] >>> Q.reciprocal().reciprocal() == Q True 
 - reduced_binary_form()[source]¶
- Find a form which is reduced in the sense that no further binary form reductions can be done to reduce the original form. - EXAMPLES: - sage: QuadraticForm(ZZ, 2, [5,5,2]).reduced_binary_form() # needs sage.symbolic ( Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 -1 ] [ * 2 ] , [ 0 -1] [ 1 1] ) - >>> from sage.all import * >>> QuadraticForm(ZZ, Integer(2), [Integer(5),Integer(5),Integer(2)]).reduced_binary_form() # needs sage.symbolic ( Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 -1 ] [ * 2 ] , <BLANKLINE> [ 0 -1] [ 1 1] ) 
 - reduced_binary_form1()[source]¶
- Reduce the form \(ax^2 + bxy+cy^2\) to satisfy the reduced condition \(|b| \le a \le c\), with \(b \ge 0\) if \(a = c\). This reduction occurs within the proper class, so all transformations are taken to have determinant 1. - EXAMPLES: - sage: QuadraticForm(ZZ, 2, [5,5,2]).reduced_binary_form1() # needs sage.symbolic ( Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 -1 ] [ * 2 ] , [ 0 -1] [ 1 1] ) - >>> from sage.all import * >>> QuadraticForm(ZZ, Integer(2), [Integer(5),Integer(5),Integer(2)]).reduced_binary_form1() # needs sage.symbolic ( Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 -1 ] [ * 2 ] , <BLANKLINE> [ 0 -1] [ 1 1] ) 
 - reduced_ternary_form__Dickson()[source]¶
- Find the unique reduced ternary form according to the conditions of Dickson’s “Studies in the Theory of Numbers”, pp164-171. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1, 1, 1]) sage: Q.reduced_ternary_form__Dickson() Traceback (most recent call last): ... NotImplementedError - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1), Integer(1)]) >>> Q.reduced_ternary_form__Dickson() Traceback (most recent call last): ... NotImplementedError 
 - representation_number_list(B)[source]¶
- Return the vector of representation numbers < B. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1,1,1,1,1]) sage: Q.representation_number_list(10) # needs sage.libs.pari [1, 16, 112, 448, 1136, 2016, 3136, 5504, 9328, 12112] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1),Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.representation_number_list(Integer(10)) # needs sage.libs.pari [1, 16, 112, 448, 1136, 2016, 3136, 5504, 9328, 12112] 
 - representation_vector_list(B, maxvectors=100000000)[source]¶
- Find all vectors \(v\) where \(Q(v) < B\). - This only works for positive definite quadratic forms. - EXAMPLES: - sage: # needs sage.libs.pari sage: Q = DiagonalQuadraticForm(ZZ, [1, 1]) sage: Q.representation_vector_list(10) [[(0, 0)], [(0, 1), (0, -1), (1, 0), (-1, 0)], [(1, 1), (-1, -1), (1, -1), (-1, 1)], [], [(0, 2), (0, -2), (2, 0), (-2, 0)], [(1, 2), (-1, -2), (1, -2), (-1, 2), (2, 1), (-2, -1), (2, -1), (-2, 1)], [], [], [(2, 2), (-2, -2), (2, -2), (-2, 2)], [(0, 3), (0, -3), (3, 0), (-3, 0)]] sage: list(map(len, _)) [1, 4, 4, 0, 4, 8, 0, 0, 4, 4] sage: Q.representation_number_list(10) [1, 4, 4, 0, 4, 8, 0, 0, 4, 4] - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1)]) >>> Q.representation_vector_list(Integer(10)) [[(0, 0)], [(0, 1), (0, -1), (1, 0), (-1, 0)], [(1, 1), (-1, -1), (1, -1), (-1, 1)], [], [(0, 2), (0, -2), (2, 0), (-2, 0)], [(1, 2), (-1, -2), (1, -2), (-1, 2), (2, 1), (-2, -1), (2, -1), (-2, 1)], [], [], [(2, 2), (-2, -2), (2, -2), (-2, 2)], [(0, 3), (0, -3), (3, 0), (-3, 0)]] >>> list(map(len, _)) [1, 4, 4, 0, 4, 8, 0, 0, 4, 4] >>> Q.representation_number_list(Integer(10)) [1, 4, 4, 0, 4, 8, 0, 0, 4, 4] 
 - scale_by_factor(c, change_value_ring_flag=False)[source]¶
- Scale the values of the quadratic form by the number \(c\), if this is possible while still being defined over its base ring. - If the flag is set to true, then this will alter the value ring to be the field of fractions of the original ring (if necessary). - INPUT: - c– a scalar in the fraction field of the value ring of the form
 - OUTPUT: a quadratic form of the same dimension - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [3,9,18,27]) sage: Q.scale_by_factor(3) Quadratic form in 4 variables over Integer Ring with coefficients: [ 9 0 0 0 ] [ * 27 0 0 ] [ * * 54 0 ] [ * * * 81 ] sage: Q.scale_by_factor(1/3) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 6 0 ] [ * * * 9 ] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(3),Integer(9),Integer(18),Integer(27)]) >>> Q.scale_by_factor(Integer(3)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 9 0 0 0 ] [ * 27 0 0 ] [ * * 54 0 ] [ * * * 81 ] >>> Q.scale_by_factor(Integer(1)/Integer(3)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 0 0 0 ] [ * 3 0 0 ] [ * * 6 0 ] [ * * * 9 ] 
 - set_number_of_automorphisms(num_autos)[source]¶
- Set the number of automorphisms to be the value given. No error checking is performed, to this may lead to erroneous results. - The fact that this result was set externally is recorded in the internal list of external initializations, accessible by the method - list_external_initializations().- OUTPUT: none - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1, 1, 1]) sage: Q.list_external_initializations() [] sage: Q.set_number_of_automorphisms(-3) sage: Q.number_of_automorphisms() -3 sage: Q.list_external_initializations() ['number_of_automorphisms'] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1), Integer(1), Integer(1)]) >>> Q.list_external_initializations() [] >>> Q.set_number_of_automorphisms(-Integer(3)) >>> Q.number_of_automorphisms() -3 >>> Q.list_external_initializations() ['number_of_automorphisms'] 
 - shimura_mass__maximal()[source]¶
- Use Shimura’s exact mass formula to compute the mass of a maximal quadratic lattice. This works for any totally real number field, but has a small technical restriction when \(n\) is odd. - OUTPUT: a rational number - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1]) sage: Q.shimura_mass__maximal() - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1)]) >>> Q.shimura_mass__maximal() 
 - short_primitive_vector_list_up_to_length(len_bound, up_to_sign_flag=False)[source]¶
- Return a list of lists of short primitive vectors \(v\), sorted by length, with \(Q(v) <\) - len_bound. The list in output \([i]\) indexes all vectors of length \(i\). If the- up_to_sign_flagis set to- True, then only one of the vectors of the pair \([v, -v]\) is listed.- Note - This processes the PARI/GP output to always give elements of type \(\ZZ\). - OUTPUT: list of lists of vectors - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.short_vector_list_up_to_length(5, True) [[(0, 0, 0, 0)], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0), (2, 0, 0, 0)]] sage: Q.short_primitive_vector_list_up_to_length(5, True) [[], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0)]] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.short_vector_list_up_to_length(Integer(5), True) [[(0, 0, 0, 0)], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0), (2, 0, 0, 0)]] >>> Q.short_primitive_vector_list_up_to_length(Integer(5), True) [[], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0)]] 
 - short_vector_list_up_to_length(len_bound, up_to_sign_flag=False)[source]¶
- Return a list of lists of short vectors \(v\), sorted by length, with \(Q(v) <\) - len_bound.- INPUT: - len_bound– bound for the length of the vectors
- up_to_sign_flag– boolean (default:- False); if set to- True, then only one of the vectors of the pair \([v, -v]\) is listed
 - OUTPUT: - A list of lists of vectors such that entry \([i]\) contains all vectors of length \(i\). - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.short_vector_list_up_to_length(3) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], []] sage: Q.short_vector_list_up_to_length(4) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], [], [(0, 1, 0, 0), (0, -1, 0, 0)]] sage: Q.short_vector_list_up_to_length(5) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], [], [(0, 1, 0, 0), (0, -1, 0, 0)], [(1, 1, 0, 0), (-1, -1, 0, 0), (1, -1, 0, 0), (-1, 1, 0, 0), (2, 0, 0, 0), (-2, 0, 0, 0)]] sage: Q.short_vector_list_up_to_length(5, True) [[(0, 0, 0, 0)], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0), (2, 0, 0, 0)]] sage: m6 = matrix(6, [2, 1, 1, 1, -1, -1, 1, 2, 1, 1, -1, -1, ....: 1, 1, 2, 0, -1, -1, 1, 1, 0, 2, 0, -1, ....: -1, -1, -1, 0, 2, 1, -1, -1, -1, -1, 1, 2]) sage: Q = QuadraticForm(m6) sage: vs = Q.short_vector_list_up_to_length(8) sage: [len(vs[i]) for i in range(len(vs))] [1, 72, 270, 720, 936, 2160, 2214, 3600] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.short_vector_list_up_to_length(Integer(3)) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], []] >>> Q.short_vector_list_up_to_length(Integer(4)) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], [], [(0, 1, 0, 0), (0, -1, 0, 0)]] >>> Q.short_vector_list_up_to_length(Integer(5)) [[(0, 0, 0, 0)], [(1, 0, 0, 0), (-1, 0, 0, 0)], [], [(0, 1, 0, 0), (0, -1, 0, 0)], [(1, 1, 0, 0), (-1, -1, 0, 0), (1, -1, 0, 0), (-1, 1, 0, 0), (2, 0, 0, 0), (-2, 0, 0, 0)]] >>> Q.short_vector_list_up_to_length(Integer(5), True) [[(0, 0, 0, 0)], [(1, 0, 0, 0)], [], [(0, 1, 0, 0)], [(1, 1, 0, 0), (1, -1, 0, 0), (2, 0, 0, 0)]] >>> m6 = matrix(Integer(6), [Integer(2), Integer(1), Integer(1), Integer(1), -Integer(1), -Integer(1), Integer(1), Integer(2), Integer(1), Integer(1), -Integer(1), -Integer(1), ... Integer(1), Integer(1), Integer(2), Integer(0), -Integer(1), -Integer(1), Integer(1), Integer(1), Integer(0), Integer(2), Integer(0), -Integer(1), ... -Integer(1), -Integer(1), -Integer(1), Integer(0), Integer(2), Integer(1), -Integer(1), -Integer(1), -Integer(1), -Integer(1), Integer(1), Integer(2)]) >>> Q = QuadraticForm(m6) >>> vs = Q.short_vector_list_up_to_length(Integer(8)) >>> [len(vs[i]) for i in range(len(vs))] [1, 72, 270, 720, 936, 2160, 2214, 3600] - The cases of - len_bound < 2led to exception or infinite runtime before.- sage: Q.short_vector_list_up_to_length(-1) [] sage: Q.short_vector_list_up_to_length(0) [] sage: Q.short_vector_list_up_to_length(1) [[(0, 0, 0, 0, 0, 0)]] - >>> from sage.all import * >>> Q.short_vector_list_up_to_length(-Integer(1)) [] >>> Q.short_vector_list_up_to_length(Integer(0)) [] >>> Q.short_vector_list_up_to_length(Integer(1)) [[(0, 0, 0, 0, 0, 0)]] - In the case of quadratic forms that are not positive definite an error is raised. - sage: QuadraticForm(matrix(2, [2, 0, 0, -2])).short_vector_list_up_to_length(3) Traceback (most recent call last): ... ValueError: Quadratic form must be positive definite in order to enumerate short vectors - >>> from sage.all import * >>> QuadraticForm(matrix(Integer(2), [Integer(2), Integer(0), Integer(0), -Integer(2)])).short_vector_list_up_to_length(Integer(3)) Traceback (most recent call last): ... ValueError: Quadratic form must be positive definite in order to enumerate short vectors - Check that PARI does not return vectors which are too long: - sage: Q = QuadraticForm(matrix(2, [72, 12, 12, 120])) sage: len_bound_pari = 2*22953421 - 2; len_bound_pari 45906840 sage: vs = list(Q.__pari__().qfminim(len_bound_pari)[2]) # long time (18s on sage.math, 2014) sage: v = vs[0]; v # long time [66, -623]~ sage: v.Vec() * Q.__pari__() * v # long time 45902280 - >>> from sage.all import * >>> Q = QuadraticForm(matrix(Integer(2), [Integer(72), Integer(12), Integer(12), Integer(120)])) >>> len_bound_pari = Integer(2)*Integer(22953421) - Integer(2); len_bound_pari 45906840 >>> vs = list(Q.__pari__().qfminim(len_bound_pari)[Integer(2)]) # long time (18s on sage.math, 2014) >>> v = vs[Integer(0)]; v # long time [66, -623]~ >>> v.Vec() * Q.__pari__() * v # long time 45902280 
 - siegel_product(u)[source]¶
- Compute the infinite product of local densities of the quadratic form for the number \(u\). - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Q.theta_series(11) 1 + 8*q + 24*q^2 + 32*q^3 + 24*q^4 + 48*q^5 + 96*q^6 + 64*q^7 + 24*q^8 + 104*q^9 + 144*q^10 + O(q^11) sage: Q.siegel_product(1) 8 sage: Q.siegel_product(2) # This one is wrong -- expect 24, and the higher powers of 2 don't work... =( 24 sage: Q.siegel_product(3) 32 sage: Q.siegel_product(5) 48 sage: Q.siegel_product(6) 96 sage: Q.siegel_product(7) 64 sage: Q.siegel_product(9) 104 sage: Q.local_density(2,1) 1 sage: M = 4; len([v for v in mrange([M,M,M,M]) if Q(v) % M == 1]) / M^3 1 sage: M = 16; len([v for v in mrange([M,M,M,M]) if Q(v) % M == 1]) / M^3 # long time (2s on sage.math, 2014) 1 sage: Q.local_density(2,2) 3/2 sage: M = 4; len([v for v in mrange([M,M,M,M]) if Q(v) % M == 2]) / M^3 3/2 sage: M = 16; len([v for v in mrange([M,M,M,M]) if Q(v) % M == 2]) / M^3 # long time (2s on sage.math, 2014) 3/2 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Q.theta_series(Integer(11)) 1 + 8*q + 24*q^2 + 32*q^3 + 24*q^4 + 48*q^5 + 96*q^6 + 64*q^7 + 24*q^8 + 104*q^9 + 144*q^10 + O(q^11) >>> Q.siegel_product(Integer(1)) 8 >>> Q.siegel_product(Integer(2)) # This one is wrong -- expect 24, and the higher powers of 2 don't work... =( 24 >>> Q.siegel_product(Integer(3)) 32 >>> Q.siegel_product(Integer(5)) 48 >>> Q.siegel_product(Integer(6)) 96 >>> Q.siegel_product(Integer(7)) 64 >>> Q.siegel_product(Integer(9)) 104 >>> Q.local_density(Integer(2),Integer(1)) 1 >>> M = Integer(4); len([v for v in mrange([M,M,M,M]) if Q(v) % M == Integer(1)]) / M**Integer(3) 1 >>> M = Integer(16); len([v for v in mrange([M,M,M,M]) if Q(v) % M == Integer(1)]) / M**Integer(3) # long time (2s on sage.math, 2014) 1 >>> Q.local_density(Integer(2),Integer(2)) 3/2 >>> M = Integer(4); len([v for v in mrange([M,M,M,M]) if Q(v) % M == Integer(2)]) / M**Integer(3) 3/2 >>> M = Integer(16); len([v for v in mrange([M,M,M,M]) if Q(v) % M == Integer(2)]) / M**Integer(3) # long time (2s on sage.math, 2014) 3/2 
 - signature()[source]¶
- Return the signature of the quadratic form, defined as: - number of positive eigenvalues \(-\) number of negative eigenvalues - of the matrix of the quadratic form. - OUTPUT: integer - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,0,0,-4,3,11,3]) sage: Q.signature() 3 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(0),Integer(0),-Integer(4),Integer(3),Integer(11),Integer(3)]) >>> Q.signature() 3 - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,-3,-4]) sage: Q.signature() 0 - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),-Integer(3),-Integer(4)]) >>> Q.signature() 0 - sage: Q = QuadraticForm(ZZ, 4, range(10)); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] sage: Q.signature() 2 - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] >>> Q.signature() 2 
 - signature_vector()[source]¶
- Return the triple \((p, n, z)\) of integers where - \(p\) = number of positive eigenvalues 
- \(n\) = number of negative eigenvalues 
- \(z\) = number of zero eigenvalues 
 - for the symmetric matrix associated to \(Q\). - OUTPUT: a triple of integers \(\geq 0\) - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,0,0,-4]) sage: Q.signature_vector() (1, 1, 2) - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(0),Integer(0),-Integer(4)]) >>> Q.signature_vector() (1, 1, 2) - sage: Q = DiagonalQuadraticForm(ZZ, [1,2,-3,-4]) sage: Q.signature_vector() (2, 2, 0) - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(2),-Integer(3),-Integer(4)]) >>> Q.signature_vector() (2, 2, 0) - sage: Q = QuadraticForm(ZZ, 4, range(10)); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] sage: Q.signature_vector() (3, 1, 0) - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(10))); Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 0 1 2 3 ] [ * 4 5 6 ] [ * * 7 8 ] [ * * * 9 ] >>> Q.signature_vector() (3, 1, 0) 
 - solve(c=0)[source]¶
- Return a vector \(x\) such that - self(x) == c.- INPUT: - c– (default: 0) a rational number
 - OUTPUT: a nonzero vector \(x\) satisfying - self(x) == c- ALGORITHM: - Uses PARI’s pari:qfsolve. Algorithm described by Jeroen Demeyer; see comments on Issue #19112 - EXAMPLES: - sage: F = DiagonalQuadraticForm(QQ, [1, -1]); F Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * -1 ] sage: F.solve() (1, 1) sage: F.solve(1) (1, 0) sage: F.solve(2) (3/2, -1/2) sage: F.solve(3) (2, -1) - >>> from sage.all import * >>> F = DiagonalQuadraticForm(QQ, [Integer(1), -Integer(1)]); F Quadratic form in 2 variables over Rational Field with coefficients: [ 1 0 ] [ * -1 ] >>> F.solve() (1, 1) >>> F.solve(Integer(1)) (1, 0) >>> F.solve(Integer(2)) (3/2, -1/2) >>> F.solve(Integer(3)) (2, -1) - sage: F = DiagonalQuadraticForm(QQ, [1, 1, 1, 1]) sage: F.solve(7) (1, 2, -1, -1) sage: F.solve() Traceback (most recent call last): ... ArithmeticError: no solution found (local obstruction at -1) - >>> from sage.all import * >>> F = DiagonalQuadraticForm(QQ, [Integer(1), Integer(1), Integer(1), Integer(1)]) >>> F.solve(Integer(7)) (1, 2, -1, -1) >>> F.solve() Traceback (most recent call last): ... ArithmeticError: no solution found (local obstruction at -1) - sage: Q = QuadraticForm(QQ, 2, [17, 94, 130]) sage: x = Q.solve(5); x (17, -6) sage: Q(x) 5 sage: Q.solve(6) Traceback (most recent call last): ... ArithmeticError: no solution found (local obstruction at 3) sage: G = DiagonalQuadraticForm(QQ, [5, -3, -2]) sage: x = G.solve(10); x (3/2, -1/2, 1/2) sage: G(x) 10 sage: F = DiagonalQuadraticForm(QQ, [1, -4]) sage: x = F.solve(); x (2, 1) sage: F(x) 0 - >>> from sage.all import * >>> Q = QuadraticForm(QQ, Integer(2), [Integer(17), Integer(94), Integer(130)]) >>> x = Q.solve(Integer(5)); x (17, -6) >>> Q(x) 5 >>> Q.solve(Integer(6)) Traceback (most recent call last): ... ArithmeticError: no solution found (local obstruction at 3) >>> G = DiagonalQuadraticForm(QQ, [Integer(5), -Integer(3), -Integer(2)]) >>> x = G.solve(Integer(10)); x (3/2, -1/2, 1/2) >>> G(x) 10 >>> F = DiagonalQuadraticForm(QQ, [Integer(1), -Integer(4)]) >>> x = F.solve(); x (2, 1) >>> F(x) 0 - sage: F = QuadraticForm(QQ, 4, [0, 0, 1, 0, 0, 0, 1, 0, 0, 0]); F Quadratic form in 4 variables over Rational Field with coefficients: [ 0 0 1 0 ] [ * 0 0 1 ] [ * * 0 0 ] [ * * * 0 ] sage: F.solve(23) (23, 0, 1, 0) - >>> from sage.all import * >>> F = QuadraticForm(QQ, Integer(4), [Integer(0), Integer(0), Integer(1), Integer(0), Integer(0), Integer(0), Integer(1), Integer(0), Integer(0), Integer(0)]); F Quadratic form in 4 variables over Rational Field with coefficients: [ 0 0 1 0 ] [ * 0 0 1 ] [ * * 0 0 ] [ * * * 0 ] >>> F.solve(Integer(23)) (23, 0, 1, 0) - Other fields besides the rationals are currently not supported: - sage: F = DiagonalQuadraticForm(GF(11), [1, 1]) sage: F.solve() Traceback (most recent call last): ... TypeError: solving quadratic forms is only implemented over QQ - >>> from sage.all import * >>> F = DiagonalQuadraticForm(GF(Integer(11)), [Integer(1), Integer(1)]) >>> F.solve() Traceback (most recent call last): ... TypeError: solving quadratic forms is only implemented over QQ 
 - split_local_cover()[source]¶
- Try to find subform of the given (positive definite quaternary) quadratic form \(Q\) of the form \[d\cdot x^2 + T(y,z,w)\]- where \(d > 0\) is as small as possible. - This is done by exhaustive search on small vectors, and then comparing the local conditions of its sum with its complementary lattice and the original quadratic form \(Q\). - OUTPUT: a - QuadraticFormover \(\ZZ\)- EXAMPLES: - sage: Q1 = DiagonalQuadraticForm(ZZ, [7,5,3]) sage: Q1.split_local_cover() # needs sage.symbolic Quadratic form in 3 variables over Integer Ring with coefficients: [ 3 0 0 ] [ * 5 0 ] [ * * 7 ] - >>> from sage.all import * >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(7),Integer(5),Integer(3)]) >>> Q1.split_local_cover() # needs sage.symbolic Quadratic form in 3 variables over Integer Ring with coefficients: [ 3 0 0 ] [ * 5 0 ] [ * * 7 ] 
 - sum_by_coefficients_with(right)[source]¶
- Return the sum (on coefficients) of two quadratic forms of the same size. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,4,10]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 4 ] [ * 10 ] sage: Q + Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 4 0 0 ] [ * 10 0 0 ] [ * * 1 4 ] [ * * * 10 ] sage: Q2 = QuadraticForm(ZZ, 2, [1,4,-10]) sage: Q.sum_by_coefficients_with(Q2) Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 8 ] [ * 0 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(4),Integer(10)]); Q Quadratic form in 2 variables over Integer Ring with coefficients: [ 1 4 ] [ * 10 ] >>> Q + Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 4 0 0 ] [ * 10 0 0 ] [ * * 1 4 ] [ * * * 10 ] >>> Q2 = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(4),-Integer(10)]) >>> Q.sum_by_coefficients_with(Q2) Quadratic form in 2 variables over Integer Ring with coefficients: [ 2 8 ] [ * 0 ] 
 - swap_variables(r, s, in_place=False)[source]¶
- Switch the variables \(x_r\) and \(x_s\) in the quadratic form (replacing the original form if the - in_placeflag is True).- INPUT: - r,- s– integers \(\geq 0\)
 - OUTPUT: - a - QuadraticForm(by default, otherwise none)- EXAMPLES: - sage: Q = QuadraticForm(ZZ, 4, range(1,11)) sage: Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] sage: Q.swap_variables(0,2) Quadratic form in 4 variables over Integer Ring with coefficients: [ 8 6 3 9 ] [ * 5 2 7 ] [ * * 1 4 ] [ * * * 10 ] sage: Q.swap_variables(0,2).swap_variables(0,2) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), range(Integer(1),Integer(11))) >>> Q Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] >>> Q.swap_variables(Integer(0),Integer(2)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 8 6 3 9 ] [ * 5 2 7 ] [ * * 1 4 ] [ * * * 10 ] >>> Q.swap_variables(Integer(0),Integer(2)).swap_variables(Integer(0),Integer(2)) Quadratic form in 4 variables over Integer Ring with coefficients: [ 1 2 3 4 ] [ * 5 6 7 ] [ * * 8 9 ] [ * * * 10 ] 
 - theta_by_cholesky(q_prec)[source]¶
- Uses the real Cholesky decomposition to compute (the \(q\)-expansion of) the theta function of the quadratic form as a power series in \(q\) with terms correct up to the power \(q^{\text{q\_prec}}\). (So its error is \(O(q^ {\text{q\_prec} + 1})\).) - REFERENCE: - Cohen’s “A Course in Computational Algebraic Number Theory” book, p 102. - EXAMPLES: - Check the sum of 4 squares form against Jacobi’s formula: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Theta = Q.theta_by_cholesky(10) sage: Theta 1 + 8*q + 24*q^2 + 32*q^3 + 24*q^4 + 48*q^5 + 96*q^6 + 64*q^7 + 24*q^8 + 104*q^9 + 144*q^10 sage: Expected = [1] + [8*sum(d for d in divisors(n) if d%4) ....: for n in range(1, 11)] sage: Expected [1, 8, 24, 32, 24, 48, 96, 64, 24, 104, 144] sage: Theta.list() == Expected True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Theta = Q.theta_by_cholesky(Integer(10)) >>> Theta 1 + 8*q + 24*q^2 + 32*q^3 + 24*q^4 + 48*q^5 + 96*q^6 + 64*q^7 + 24*q^8 + 104*q^9 + 144*q^10 >>> Expected = [Integer(1)] + [Integer(8)*sum(d for d in divisors(n) if d%Integer(4)) ... for n in range(Integer(1), Integer(11))] >>> Expected [1, 8, 24, 32, 24, 48, 96, 64, 24, 104, 144] >>> Theta.list() == Expected True - Check the form \(x^2 + 3y^2 + 5z^2 + 7w^2\) represents everything except 2 and 22.: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Theta = Q.theta_by_cholesky(50) sage: Theta_list = Theta.list() sage: [m for m in range(len(Theta_list)) if Theta_list[m] == 0] [2, 22] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Theta = Q.theta_by_cholesky(Integer(50)) >>> Theta_list = Theta.list() >>> [m for m in range(len(Theta_list)) if Theta_list[m] == Integer(0)] [2, 22] 
 - theta_by_pari(Max, var_str='q', safe_flag=True)[source]¶
- Use PARI/GP to compute the theta function as a power series (or vector) up to the precision \(O(q^{Max})\). This also caches the result for future computations. - If - var_str=- '', then we return a vector \(v\) where- v[i]counts the number of vectors of length \(i\).- The - safe_flagallows us to select whether we want a copy of the output, or the original output. It is only meaningful when a vector is returned, otherwise a copy is automatically made in creating the power series. By default- safe_flag=True, so we return a copy of the cached information. If this is set to- False, then the routine is much faster but the return values are vulnerable to being corrupted by the user.- INPUT: - Max– integer \(\geq 0\)
- var_str– string
 - OUTPUT: a power series or a vector - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1,1,1]) sage: Prec = 100 sage: compute = Q.theta_by_pari(Prec, '') # needs sage.libs.pari sage: exact = [1] + [8 * sum([d for d in divisors(i) if d % 4 != 0]) # needs sage.libs.pari ....: for i in range(1, Prec)] sage: compute == exact # needs sage.libs.pari True - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1),Integer(1),Integer(1)]) >>> Prec = Integer(100) >>> compute = Q.theta_by_pari(Prec, '') # needs sage.libs.pari >>> exact = [Integer(1)] + [Integer(8) * sum([d for d in divisors(i) if d % Integer(4) != Integer(0)]) # needs sage.libs.pari ... for i in range(Integer(1), Prec)] >>> compute == exact # needs sage.libs.pari True 
 - theta_series(Max=10, var_str='q', safe_flag=True)[source]¶
- Compute the theta series as a power series in the variable given in - var_str(which defaults to- 'q'), up to the specified precision \(O(q^{Max})\).- This uses the PARI/GP function pari:qfrep, wrapped by the theta_by_pari() method. This caches the result for future computations. - The - safe_flagallows us to select whether we want a copy of the output, or the original output. It is only meaningful when a vector is returned, otherwise a copy is automatically made in creating the power series. By default- safe_flag=- True, so we return a copy of the cached information. If this is set to- False, then the routine is much faster but the return values are vulnerable to being corrupted by the user.- Todo - Allow the option - Max='mod_form'to give enough coefficients to ensure we determine the theta series as a modular form. This is related to the Sturm bound, but we will need to be careful about this (particularly for half-integral weights!).- EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q.theta_series() # needs sage.libs.pari 1 + 2*q + 2*q^3 + 6*q^4 + 2*q^5 + 4*q^6 + 6*q^7 + 8*q^8 + 14*q^9 + O(q^10) sage: Q.theta_series(25) # needs sage.libs.pari 1 + 2*q + 2*q^3 + 6*q^4 + 2*q^5 + 4*q^6 + 6*q^7 + 8*q^8 + 14*q^9 + 4*q^10 + 12*q^11 + 18*q^12 + 12*q^13 + 12*q^14 + 8*q^15 + 34*q^16 + 12*q^17 + 8*q^18 + 32*q^19 + 10*q^20 + 28*q^21 + 16*q^23 + 44*q^24 + O(q^25) - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q.theta_series() # needs sage.libs.pari 1 + 2*q + 2*q^3 + 6*q^4 + 2*q^5 + 4*q^6 + 6*q^7 + 8*q^8 + 14*q^9 + O(q^10) >>> Q.theta_series(Integer(25)) # needs sage.libs.pari 1 + 2*q + 2*q^3 + 6*q^4 + 2*q^5 + 4*q^6 + 6*q^7 + 8*q^8 + 14*q^9 + 4*q^10 + 12*q^11 + 18*q^12 + 12*q^13 + 12*q^14 + 8*q^15 + 34*q^16 + 12*q^17 + 8*q^18 + 32*q^19 + 10*q^20 + 28*q^21 + 16*q^23 + 44*q^24 + O(q^25) 
 - theta_series_degree_2(Q, prec)[source]¶
- Compute the theta series of degree 2 for the quadratic form \(Q\). - INPUT: - prec– integer
 - OUTPUT: dictionary, where: - keys are \({\rm GL}_2(\ZZ)\)-reduced binary quadratic forms (given as triples of coefficients) 
- values are coefficients 
 - EXAMPLES: - sage: # needs sage.symbolic sage: Q2 = QuadraticForm(ZZ, 4, [1,1,1,1, 1,0,0, 1,0, 1]) sage: S = Q2.theta_series_degree_2(10) sage: S[(0,0,2)] 24 sage: S[(1,0,1)] 144 sage: S[(1,1,1)] 192 - >>> from sage.all import * >>> # needs sage.symbolic >>> Q2 = QuadraticForm(ZZ, Integer(4), [Integer(1),Integer(1),Integer(1),Integer(1), Integer(1),Integer(0),Integer(0), Integer(1),Integer(0), Integer(1)]) >>> S = Q2.theta_series_degree_2(Integer(10)) >>> S[(Integer(0),Integer(0),Integer(2))] 24 >>> S[(Integer(1),Integer(0),Integer(1))] 144 >>> S[(Integer(1),Integer(1),Integer(1))] 192 - AUTHORS: - Gonzalo Tornaria (2010-03-23) 
 - REFERENCE: - Raum, Ryan, Skoruppa, Tornaria, ‘On Formal Siegel Modular Forms’ (preprint) 
 
 - vectors_by_length(bound)[source]¶
- Return a list of short vectors together with their values. - This is a naive algorithm which uses the Cholesky decomposition, but does not use the LLL-reduction algorithm. - INPUT: - bound– integer \(\geq 0\)
 - OUTPUT: - a list - Lof length (- bound+ 1) whose entry- L[i]is a list of all vectors of length \(i\).
 - REFERENCES: - This is a slightly modified version of Cohn’s Algorithm 2.7.5 in “A Course in Computational Number Theory”, with the increment step moved around and slightly re-indexed to allow clean looping. - Note - We could speed this up for very skew matrices by using LLL first, and then changing coordinates back, but for our purposes the simpler method is efficient enough. - EXAMPLES: - sage: Q = DiagonalQuadraticForm(ZZ, [1,1]) sage: Q.vectors_by_length(5) # needs sage.symbolic [[[0, 0]], [[0, -1], [-1, 0]], [[-1, -1], [1, -1]], [], [[0, -2], [-2, 0]], [[-1, -2], [1, -2], [-2, -1], [2, -1]]] - >>> from sage.all import * >>> Q = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(1)]) >>> Q.vectors_by_length(Integer(5)) # needs sage.symbolic [[[0, 0]], [[0, -1], [-1, 0]], [[-1, -1], [1, -1]], [], [[0, -2], [-2, 0]], [[-1, -2], [1, -2], [-2, -1], [2, -1]]] - sage: Q1 = DiagonalQuadraticForm(ZZ, [1,3,5,7]) sage: Q1.vectors_by_length(5) # needs sage.symbolic [[[0, 0, 0, 0]], [[-1, 0, 0, 0]], [], [[0, -1, 0, 0]], [[-1, -1, 0, 0], [1, -1, 0, 0], [-2, 0, 0, 0]], [[0, 0, -1, 0]]] - >>> from sage.all import * >>> Q1 = DiagonalQuadraticForm(ZZ, [Integer(1),Integer(3),Integer(5),Integer(7)]) >>> Q1.vectors_by_length(Integer(5)) # needs sage.symbolic [[[0, 0, 0, 0]], [[-1, 0, 0, 0]], [], [[0, -1, 0, 0]], [[-1, -1, 0, 0], [1, -1, 0, 0], [-2, 0, 0, 0]], [[0, 0, -1, 0]]] - sage: Q = QuadraticForm(ZZ, 4, [1,1,1,1, 1,0,0, 1,0, 1]) sage: list(map(len, Q.vectors_by_length(2))) # needs sage.symbolic [1, 12, 12] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), [Integer(1),Integer(1),Integer(1),Integer(1), Integer(1),Integer(0),Integer(0), Integer(1),Integer(0), Integer(1)]) >>> list(map(len, Q.vectors_by_length(Integer(2)))) # needs sage.symbolic [1, 12, 12] - sage: Q = QuadraticForm(ZZ, 4, [1,-1,-1,-1, 1,0,0, 4,-3, 4]) sage: list(map(len, Q.vectors_by_length(3))) # needs sage.symbolic [1, 3, 0, 3] - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(4), [Integer(1),-Integer(1),-Integer(1),-Integer(1), Integer(1),Integer(0),Integer(0), Integer(4),-Integer(3), Integer(4)]) >>> list(map(len, Q.vectors_by_length(Integer(3)))) # needs sage.symbolic [1, 3, 0, 3] 
 - xi(p)[source]¶
- Return the value of the genus characters Xi_p… which may be missing one character. We allow -1 as a prime. - REFERENCES: - Dickson’s “Studies in the Theory of Numbers” - EXAMPLES: - sage: Q1 = QuadraticForm(ZZ, 3, [1, 1, 1, 14, 3, 14]) sage: Q2 = QuadraticForm(ZZ, 3, [2, -1, 0, 2, 0, 50]) sage: [Q1.omega(), Q2.omega()] [5, 5] sage: [Q1.hasse_invariant(5), # equivalent over Q_5 # needs sage.libs.pari ....: Q2.hasse_invariant(5)] [1, 1] sage: [Q1.xi(5), Q2.xi(5)] # not equivalent over Z_5 # needs sage.libs.pari [1, -1] - >>> from sage.all import * >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(1), Integer(1), Integer(14), Integer(3), Integer(14)]) >>> Q2 = QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(1), Integer(0), Integer(2), Integer(0), Integer(50)]) >>> [Q1.omega(), Q2.omega()] [5, 5] >>> [Q1.hasse_invariant(Integer(5)), # equivalent over Q_5 # needs sage.libs.pari ... Q2.hasse_invariant(Integer(5))] [1, 1] >>> [Q1.xi(Integer(5)), Q2.xi(Integer(5))] # not equivalent over Z_5 # needs sage.libs.pari [1, -1] 
 - xi_rec(p)[source]¶
- Return Xi(\(p\)) for the reciprocal form. - EXAMPLES: - sage: # needs sage.libs.pari sage: Q1 = QuadraticForm(ZZ, 3, [1, 1, 1, 14, 3, 14]) sage: Q2 = QuadraticForm(ZZ, 3, [2, -1, 0, 2, 0, 50]) sage: [Q1.clifford_conductor(), # equivalent over Q ....: Q2.clifford_conductor()] [3, 3] sage: Q1.is_locally_equivalent_to(Q2) # not in the same genus False sage: [Q1.delta(), Q2.delta()] [480, 480] sage: factor(480) 2^5 * 3 * 5 sage: list(map(Q1.xi_rec, [-1,2,3,5])) [-1, -1, -1, 1] sage: list(map(Q2.xi_rec, [-1,2,3,5])) [-1, -1, -1, -1] - >>> from sage.all import * >>> # needs sage.libs.pari >>> Q1 = QuadraticForm(ZZ, Integer(3), [Integer(1), Integer(1), Integer(1), Integer(14), Integer(3), Integer(14)]) >>> Q2 = QuadraticForm(ZZ, Integer(3), [Integer(2), -Integer(1), Integer(0), Integer(2), Integer(0), Integer(50)]) >>> [Q1.clifford_conductor(), # equivalent over Q ... Q2.clifford_conductor()] [3, 3] >>> Q1.is_locally_equivalent_to(Q2) # not in the same genus False >>> [Q1.delta(), Q2.delta()] [480, 480] >>> factor(Integer(480)) 2^5 * 3 * 5 >>> list(map(Q1.xi_rec, [-Integer(1),Integer(2),Integer(3),Integer(5)])) [-1, -1, -1, 1] >>> list(map(Q2.xi_rec, [-Integer(1),Integer(2),Integer(3),Integer(5)])) [-1, -1, -1, -1] 
 
- sage.quadratic_forms.quadratic_form.is_QuadraticForm(Q)[source]¶
- Determine if the object - Qis an element of the- QuadraticFormclass.- This function is deprecated. - EXAMPLES: - sage: Q = QuadraticForm(ZZ, 2, [1,2,3]) sage: from sage.quadratic_forms.quadratic_form import is_QuadraticForm sage: is_QuadraticForm(Q) doctest:...: DeprecationWarning: the function is_QuadraticForm is deprecated; use isinstance(x, sage.quadratic_forms.quadratic_form.QuadraticForm) instead... True sage: is_QuadraticForm(2) False - >>> from sage.all import * >>> Q = QuadraticForm(ZZ, Integer(2), [Integer(1),Integer(2),Integer(3)]) >>> from sage.quadratic_forms.quadratic_form import is_QuadraticForm >>> is_QuadraticForm(Q) doctest:...: DeprecationWarning: the function is_QuadraticForm is deprecated; use isinstance(x, sage.quadratic_forms.quadratic_form.QuadraticForm) instead... True >>> is_QuadraticForm(Integer(2)) False 
- sage.quadratic_forms.quadratic_form.quadratic_form_from_invariants(F, rk, det, P, sminus)[source]¶
- Return a rational quadratic form with given invariants. - INPUT: - F– the base field; currently only- QQis allowed
- rk– integer; the rank
- det– rational; the determinant
- P– list of primes where Cassel’s Hasse invariant is negative
- sminus– integer; the number of negative eigenvalues of any Gram matrix
 - OUTPUT: a quadratic form with the specified invariants - Let \((a_1, \ldots, a_n)\) be the gram marix of a regular quadratic space. Then Cassel’s Hasse invariant is defined as \[\prod_{i<j} (a_i,a_j),\]- where \((a_i,a_j)\) denotes the Hilbert symbol. - ALGORITHM: - We follow [Kir2016]. - EXAMPLES: - sage: P = [3,5] sage: q = quadratic_form_from_invariants(QQ,2,-15,P,1); q # needs sage.rings.padics Quadratic form in 2 variables over Rational Field with coefficients: [ 5 0 ] [ * -3 ] sage: all(q.hasse_invariant(p) == -1 for p in P) # needs sage.rings.padics True - >>> from sage.all import * >>> P = [Integer(3),Integer(5)] >>> q = quadratic_form_from_invariants(QQ,Integer(2),-Integer(15),P,Integer(1)); q # needs sage.rings.padics Quadratic form in 2 variables over Rational Field with coefficients: [ 5 0 ] [ * -3 ] >>> all(q.hasse_invariant(p) == -Integer(1) for p in P) # needs sage.rings.padics True