Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
""" `p`-Adic Generic Nodes
This file contains a bunch of intermediate classes for the `p`-adic parents, allowing a function to be implemented at the right level of generality.
AUTHORS:
- David Roe """
#***************************************************************************** # Copyright (C) 2007-2013 David Roe <roed.math@gmail.com> # William Stein <wstein@gmail.com> # # Distributed under the terms of the GNU General Public License (GPL) # as published by the Free Software Foundation; either version 2 of # the License, or (at your option) any later version. # # http://www.gnu.org/licenses/ #***************************************************************************** from six import iteritems
from sage.rings.padics.local_generic import LocalGeneric from sage.rings.padics.padic_generic import pAdicGeneric from sage.rings.ring import EuclideanDomain, Field from sage.rings.padics.padic_base_generic import pAdicBaseGeneric from sage.rings.integer_ring import ZZ from sage.rings.rational_field import QQ from sage.rings.infinity import infinity, SignError from .lattice_precision import PrecisionLattice, PrecisionModule from .padic_lattice_element import pAdicLatticeElement, pAdicLatticeCapElement, pAdicLatticeFloatElement
class CappedAbsoluteGeneric(LocalGeneric): def is_capped_absolute(self): """ Returns whether this `p`-adic ring bounds precision in a capped absolute fashion.
The absolute precision of an element is the power of `p` modulo which that element is defined. In a capped absolute ring, the absolute precision of elements are bounded by a constant depending on the ring.
EXAMPLES::
sage: R = ZpCA(5, 15) sage: R.is_capped_absolute() True sage: R(5^7) 5^7 + O(5^15) sage: S = Zp(5, 15) sage: S.is_capped_absolute() False sage: S(5^7) 5^7 + O(5^22) """
def _prec_type(self): """ Returns the precision handling type.
EXAMPLES::
sage: ZpCA(5)._prec_type() 'capped-abs' """
class CappedRelativeGeneric(LocalGeneric): def is_capped_relative(self): """ Returns whether this `p`-adic ring bounds precision in a capped relative fashion.
The relative precision of an element is the power of p modulo which the unit part of that element is defined. In a capped relative ring, the relative precision of elements are bounded by a constant depending on the ring.
EXAMPLES::
sage: R = ZpCA(5, 15) sage: R.is_capped_relative() False sage: R(5^7) 5^7 + O(5^15) sage: S = Zp(5, 15) sage: S.is_capped_relative() True sage: S(5^7) 5^7 + O(5^22) """
def _prec_type(self): """ Returns the precision handling type.
EXAMPLES::
sage: Zp(5)._prec_type() 'capped-rel' """
class FixedModGeneric(LocalGeneric): def is_fixed_mod(self): """ Returns whether this `p`-adic ring bounds precision in a fixed modulus fashion.
The absolute precision of an element is the power of p modulo which that element is defined. In a fixed modulus ring, the absolute precision of every element is defined to be the precision cap of the parent. This means that some operations, such as division by `p`, don't return a well defined answer.
EXAMPLES::
sage: R = ZpFM(5,15) sage: R.is_fixed_mod() True sage: R(5^7,absprec=9) 5^7 + O(5^15) sage: S = ZpCA(5, 15) sage: S.is_fixed_mod() False sage: S(5^7,absprec=9) 5^7 + O(5^9) """
def _prec_type(self): """ Returns the precision handling type.
EXAMPLES::
sage: ZpFM(5)._prec_type() 'fixed-mod' """
class FloatingPointGeneric(LocalGeneric): def is_floating_point(self): """ Returns whether this `p`-adic ring uses a floating point precision model.
Elements in the floating point model are stored by giving a valuation and a unit part. Arithmetic is done where the unit part is truncated modulo a fixed power of the uniformizer, stored in the precision cap of the parent.
EXAMPLES::
sage: R = ZpFP(5,15) sage: R.is_floating_point() True sage: R(5^7,absprec=9) 5^7 sage: S = ZpCR(5,15) sage: S.is_floating_point() False sage: S(5^7,absprec=9) 5^7 + O(5^9) """
def _prec_type(self): """ Returns the precision handling type.
EXAMPLES::
sage: ZpFP(5)._prec_type() 'floating-point' """
def _test_distributivity(self, **options): r""" Test the distributivity of `*` on `+` on (not necessarily all) elements of this set.
p-adic floating point rings only satisfy distributivity up to a precision that depends on the elements.
INPUT:
- ``options`` -- any keyword arguments accepted by :meth:`_tester`
EXAMPLES:
By default, this method runs the tests only on the elements returned by ``self.some_elements()``::
sage: R = ZpFP(5,3) sage: R.some_elements() [0, 1, 5, 1 + 3*5 + 3*5^2, 5 + 4*5^2 + 4*5^3] sage: R._test_distributivity()
However, the elements tested can be customized with the ``elements`` keyword argument::
sage: R._test_distributivity(elements=[R(0),~R(0),R(42)])
See the documentation for :class:`TestSuite` for more information. """ x.valuation() + y.valuation() + (x * y).precision_relative(), x.valuation() + z.valuation() + (x * z).precision_relative()) else: # only check left distributivity, since multiplication commutative
def _test_additive_associativity(self, **options): r""" Test associativity for (not necessarily all) elements of this additive semigroup.
INPUT:
- ``options`` -- any keyword arguments accepted by :meth:`_tester`
EXAMPLES:
By default, this method tests only the elements returned by ``self.some_elements()``::
sage: R = QpFP(7,3) sage: R._test_additive_associativity()
However, the elements tested can be customized with the ``elements`` keyword argument::
sage: R._test_additive_associativity(elements = [R(0), ~R(0), R(42)])
See the documentation for :class:`TestSuite` for more information. """
class FloatingPointRingGeneric(FloatingPointGeneric): pass class FloatingPointFieldGeneric(FloatingPointGeneric):#, sage.rings.ring.Field): pass class CappedRelativeRingGeneric(CappedRelativeGeneric): pass class CappedRelativeFieldGeneric(CappedRelativeGeneric):#, sage.rings.ring.Field): pass
class pAdicLatticeGeneric(pAdicGeneric): r""" An implementation of the `p`-adic rationals with lattice precision.
INPUT:
- `p` -- the underlying prime number
- ``prec`` -- the precision
- ``subtype`` -- either ``"cap"`` or ``"float"``, specifying the precision model used for tracking precision
- ``label`` -- a string or ``None`` (default: ``None``)
TESTS::
sage: R = ZpLC(17) # indirect doctest doctest:...: FutureWarning: This class/method/function is marked as experimental. It, its functionality or its interface might change without a formal deprecation. See http://trac.sagemath.org/23505 for details. sage: R._prec_type() 'lattice-cap'
sage: R = ZpLF(17) # indirect doctest sage: R._prec_type() 'lattice-float'
sage: R = QpLC(17) # indirect doctest sage: R._prec_type() 'lattice-cap'
sage: R = QpLF(17) # indirect doctest sage: R._prec_type() 'lattice-float' """ def __init__(self, p, prec, print_mode, names, label=None): """ Initialization.
TESTS::
sage: R = ZpLC(17) # indirect doctest sage: R._prec_type() 'lattice-cap' sage: R._subtype 'cap'
sage: R = ZpLF(17) # indirect doctest sage: R._prec_type() 'lattice-float' sage: R._subtype 'float' """ else: # We do not use the standard attribute element_class # because we need to be careful with precision # Instead we implement _element_constructor_ (cf below) else: raise ValueError("subtype must be either 'cap' or 'float'")
def _prec_type(self): """ Return the precision handling type.
EXAMPLES::
sage: ZpLC(5)._prec_type() 'lattice-cap' """
def is_lattice_prec(self): """ Returns whether this `p`-adic ring bounds precision using a lattice model.
In lattice precision, relationships between elements are stored in a precision object of the parent, which allows for optimal precision tracking at the cost of increased memory usage and runtime.
EXAMPLES::
sage: R = ZpCR(5, 15) sage: R.is_lattice_prec() False sage: x = R(25, 8) sage: x - x O(5^8) sage: S = ZpLC(5, 15) sage: S.is_lattice_prec() True sage: x = S(25, 8) sage: x - x O(5^30) """
def precision_cap(self): """ Return the relative precision cap for this ring if it is finite. Otherwise return the absolute precision cap.
EXAMPLES::
sage: R = ZpLC(3) sage: R.precision_cap() 20 sage: R.precision_cap_relative() 20
sage: R = ZpLC(3, prec=(infinity,20)) sage: R.precision_cap() 20 sage: R.precision_cap_relative() +Infinity sage: R.precision_cap_absolute() 20
.. SEEALSO::
:meth:`precision_cap_relative`, :meth:`precision_cap_absolute` """ else:
def _precision_cap(self): """ Return the pair of precisions (for ``lattice-cap``) or the relative precision cap (for ``lattice-float``).
EXAMPLES::
sage: R = ZpLC(11, (27,37)) sage: R._precision_cap() (27, 37) sage: R = ZpLF(11, 14) sage: R._precision_cap() 14 """ else:
def precision_cap_relative(self): """ Return the relative precision cap for this ring.
EXAMPLES::
sage: R = ZpLC(3) sage: R.precision_cap_relative() 20
sage: R = ZpLC(3, prec=(infinity,20)) sage: R.precision_cap_relative() +Infinity
.. SEEALSO::
:meth:`precision_cap`, :meth:`precision_cap_absolute` """
def precision_cap_absolute(self): """ Return the absolute precision cap for this ring.
EXAMPLES::
sage: R = ZpLC(3) sage: R.precision_cap_absolute() 40
sage: R = ZpLC(3, prec=(infinity,20)) sage: R.precision_cap_absolute() 20
.. SEEALSO::
:meth:`precision_cap`, :meth:`precision_cap_relative` """
def precision(self): """ Return the lattice precision object attached to this parent.
EXAMPLES::
sage: R = ZpLC(5, label='precision') sage: R.precision() Precision lattice on 0 objects (label: precision)
sage: x = R(1, 10); y = R(1, 5) sage: R.precision() Precision lattice on 2 objects (label: precision)
.. SEEALSO::
:class:`sage.rings.padics.lattice_precision.PrecisionLattice` """
def label(self): """ Return the label of this parent.
NOTE:
Labels can be used to distinguish between parents with the same defining data.
They are useful in the lattice precision framework in order to limit the size of the lattice modeling the precision (which is roughly the number of elements having this parent).
Elements of a parent with some label do not coerce to a parent with a different label. However conversions are allowed.
EXAMPLES:
sage: R = ZpLC(5) sage: R.label() # no label by default
sage: R = ZpLC(5, label='mylabel') sage: R.label() 'mylabel'
Labels are typically useful to isolate computations. For example, assume that we first want to do some calculations with matrices::
sage: R = ZpLC(5, label='matrices') sage: M = random_matrix(R, 4, 4) sage: d = M.determinant()
Now, if we want to do another unrelated computation, we can use a different label::
sage: R = ZpLC(5, label='polynomials') sage: S.<x> = PolynomialRing(R) sage: P = (x-1)*(x-2)*(x-3)*(x-4)*(x-5)
Without labels, the software would have modeled the precision on the matrices and on the polynomials using the same lattice (manipulating a lattice of higher dimension can have a significant impact on performance). """
def _element_constructor_(self, x, prec=None): """ Create an element of this parent.
INPUT:
- ``x``: the datum from which the element is created
- ``prec`` -- an integer or ``None`` (the default); the absolute precision of the created element
NOTE:
This function tries to be sharp on precision as much as possible. For instance, if the datum ``x`` is itself an element of the same parent, the software remembers that the created element is actually equal to ``x`` (at infinite precision)::
sage: R = ZpLC(2, prec=(infinity,50)) sage: x = R(1, 10); x 1 + O(2^10) sage: y = R(x) # indirect doctest sage: y 1 + O(2^10) sage: x - y O(2^50) """ # We first try the _copy method which is sharp on precision return x.add_bigoh(prec) else:
def convert_multiple(self, *elts): """ Convert a list of elements to this parent.
NOTE:
This function tries to be sharp on precision as much as possible. In particular, if the precision of the input elements are handled by a lattice, diffused digits of precision are preserved during the conversion.
EXAMPLES::
sage: R = ZpLC(2) sage: x = R(1, 10); y = R(1, 5) sage: x,y = x+y, x-y
Remark that the pair `(x,y)` has diffused digits of precision::
sage: x 2 + O(2^5) sage: y O(2^5) sage: x + y 2 + O(2^11)
sage: R.precision().diffused_digits([x,y]) 6
As a consequence, if we convert ``x`` and ``y`` separately, we loose some precision::
sage: R2 = ZpLC(2, label='copy') sage: x2 = R2(x); y2 = R2(y) sage: x2 2 + O(2^5) sage: y2 O(2^5) sage: x2 + y2 2 + O(2^5)
sage: R2.precision().diffused_digits([x2,y2]) 0
On the other hand, this issue dissapears when we use multiple conversion::
sage: x2,y2 = R2.convert_multiple(x,y) sage: x2 + y2 2 + O(2^11)
sage: R2.precision().diffused_digits([x2,y2]) 6 """
# We sort elements by precision lattice indices[idx].append(i) else: raise TypeError("conversion between different p-adic rings not supported") else: else: elt_other.append(x)
# We create the elements # First the elements with precision lattice # Here, we use the _copy method in order # to be sharp on precision for x in L: y = x._copy(parent=self) for i in indices[id(x)]: ans[i] = y else: except PrecisionError: raise NotImplementedError("multiple conversion of a set of variables for which the module precision is not a lattice is not implemented yet") # Now the other elements y = self._element_class(self, x) for i in indices[id(x)]: ans[i] = y
# We return the created elements
def is_pAdicRing(R): """ Returns ``True`` if and only if ``R`` is a `p`-adic ring (not a field).
EXAMPLES::
sage: is_pAdicRing(Zp(5)) True sage: is_pAdicRing(RR) False """
class pAdicRingGeneric(pAdicGeneric, EuclideanDomain): def is_field(self, proof = True): """ Returns whether this ring is actually a field, ie ``False``.
EXAMPLES::
sage: Zp(5).is_field() False """
def krull_dimension(self): r""" Returns the Krull dimension of self, i.e. 1
INPUT:
- self -- a `p`-adic ring
OUTPUT:
- the Krull dimension of self. Since self is a `p`-adic ring, this is 1.
EXAMPLES::
sage: Zp(5).krull_dimension() 1 """
def is_pAdicField(R): """ Returns ``True`` if and only if ``R`` is a `p`-adic field.
EXAMPLES::
sage: is_pAdicField(Zp(17)) False sage: is_pAdicField(Qp(17)) True """
class pAdicFieldGeneric(pAdicGeneric, Field): pass
#def class_field(self, group=None, map=None, generators=None): # raise NotImplementedError
#def composite(self, subfield1, subfield2): # raise NotImplementedError
#def norm_equation(self): # raise NotImplementedError
#def norm_group(self): # raise NotImplementedError
#def norm_group_discriminant(self, group=None, map=None, generators=None): # raise NotImplementedError
#def number_of_extensions(self, degree, discriminant=None, e=None, f=None): # raise NotImplementedError
#def list_of_extensions(self, degree, discriminant=None, e=None, f=None): # raise NotImplementedError
#def subfield(self, list): # raise NotImplementedError
#def subfield_lattice(self): # raise NotImplementedError
#def subfields_of_degree(self, n): # raise NotImplementedError
class pAdicFixedModRingGeneric(pAdicRingGeneric, FixedModGeneric): pass class pAdicCappedAbsoluteRingGeneric(pAdicRingGeneric, CappedAbsoluteGeneric): pass class pAdicCappedRelativeRingGeneric(pAdicRingGeneric, CappedRelativeRingGeneric): pass class pAdicCappedRelativeFieldGeneric(pAdicFieldGeneric, CappedRelativeFieldGeneric): pass class pAdicFloatingPointRingGeneric(pAdicRingGeneric, FloatingPointRingGeneric): pass class pAdicFloatingPointFieldGeneric(pAdicFieldGeneric, FloatingPointFieldGeneric): pass
class pAdicRingBaseGeneric(pAdicBaseGeneric, pAdicRingGeneric): def construction(self): """ Returns the functorial construction of self, namely, completion of the rational numbers with respect a given prime.
Also preserves other information that makes this field unique (e.g. precision, rounding, print mode).
EXAMPLES::
sage: K = Zp(17, 8, print_mode='val-unit', print_sep='&') sage: c, L = K.construction(); L Integer Ring sage: c(L) 17-adic Ring with capped relative precision 8 sage: K == c(L) True
TESTS::
sage: R = ZpLC(13,(31,41)) sage: R._precision_cap() (31, 41) sage: F, Z = R.construction() sage: S = F(Z) sage: S._precision_cap() (31, 41) """
def random_element(self, algorithm='default'): r""" Returns a random element of self, optionally using the algorithm argument to decide how it generates the element. Algorithms currently implemented:
- default: Choose `a_i`, `i >= 0`, randomly between `0` and `p-1` until a nonzero choice is made. Then continue choosing `a_i` randomly between `0` and `p-1` until we reach precision_cap, and return `\sum a_i p^i`.
EXAMPLES::
sage: Zp(5,6).random_element() 3 + 3*5 + 2*5^2 + 3*5^3 + 2*5^4 + 5^5 + O(5^6) sage: ZpCA(5,6).random_element() 4*5^2 + 5^3 + O(5^6) sage: ZpFM(5,6).random_element() 2 + 4*5^2 + 2*5^4 + 5^5 + O(5^6) """ else: else: raise NotImplementedError("Don't know %s algorithm"%algorithm)
#def unit_group(self): # raise NotImplementedError
#def unit_group_gens(self): # raise NotImplementedError
#def principal_unit_group(self): # raise NotImplementedError
class pAdicFieldBaseGeneric(pAdicBaseGeneric, pAdicFieldGeneric): def composite(self, subfield1, subfield2): r""" Returns the composite of two subfields of self, i.e., the largest subfield containing both
INPUT:
- ``self`` -- a `p`-adic field - ``subfield1`` -- a subfield - ``subfield2`` -- a subfield
OUTPUT:
- the composite of subfield1 and subfield2
EXAMPLES::
sage: K = Qp(17); K.composite(K, K) is K True """ #should be overridden for extension fields raise ValueError("Arguments must be subfields of self.")
def subfields_of_degree(self, n): r""" Returns the number of subfields of self of degree `n`
INPUT:
- ``self`` -- a `p`-adic field - ``n`` -- an integer
OUTPUT:
- integer -- the number of subfields of degree ``n`` over self.base_ring()
EXAMPLES::
sage: K = Qp(17) sage: K.subfields_of_degree(1) 1 """ else: return 0
def subfield(self, list): r""" Returns the subfield generated by the elements in list
INPUT:
- ``self`` -- a `p`-adic field - ``list`` -- a list of elements of ``self``
OUTPUT:
- the subfield of ``self`` generated by the elements of list
EXAMPLES::
sage: K = Qp(17); K.subfield([K(17), K(1827)]) is K True """ raise TypeError("Members of the list of generators must be elements of self.")
def construction(self): """ Returns the functorial construction of ``self``, namely, completion of the rational numbers with respect a given prime.
Also preserves other information that makes this field unique (e.g. precision, rounding, print mode).
EXAMPLES::
sage: K = Qp(17, 8, print_mode='val-unit', print_sep='&') sage: c, L = K.construction(); L Rational Field sage: c(L) 17-adic Field with capped relative precision 8 sage: K == c(L) True
TESTS::
sage: R = QpLC(13,(31,41)) sage: R._precision_cap() (31, 41) sage: F, Z = R.construction() sage: S = F(Z) sage: S._precision_cap() (31, 41) """ |