Vincent Delecroix wrote: > That does not prevent us to normalize for let say: __repr__() , > numerator(), denominator(). Is there any reasonable rule to choose > when to and when not to normalize? For QQ itself, GMP does normalize > all the time.
Just to be certain that we are talking about the same thing: Currently, elements of fraction fields are always reduced (except over inexact rings) in the sense of dividing out by the gcd of the numerator and the denominator. But no further normalization is done, and in particular, when the numerator and denominator are polynomials, their contents or leading coefficients are not normalized. There is a ticket needing review that proposes to normalize the leading coefficient of the denominator to one: https://trac.sagemath.org/ticket/16268 Actually, I first tried implementing a version that would systematically clear denominators inside the numerator and denominator of the fraction, but I didn't manage to make it reasonably fast, even with a dedicated class for polynomials over fraction fields (with a single denominator, similar to the representation of polynomials over ℚ). While I'm not saying it can't be done, in the short term, the version at #16268 is a lot simpler and already fixes a number of issues! Regarding your question: Having a variant of numerator()/denominator() that works recursively would definitely be useful, and yes, it may be nice to use it in __repr__(). At the same time, I think we do want to keep programmatic access to the “true” numerator and denominator as defined in the data structure. In case you want to experiment with the idea, here is a (perhaps a bit naive, improvements welcome!) implementation of something similar that I needed for something I'm working on. It is intended to be used with #16268 and #23909. def _my_lcm(elts): # non monic l = ZZ.one() for p in elts: l *= (p//p.gcd(l)) return l def _clear_denominators_1(elt, dom): num, den = elt.numerator(), elt.denominator() base = dom.base_ring() if isinstance(base, FractionField_generic): numnum, numden = clear_denominators(num, base) dennum, denden = clear_denominators(den, base) newdom = dom.change_ring(base.ring()) numnum = newdom(numnum) dennum = newdom(dennum) gnum = numnum.gcd(dennum) numnum, dennum = numnum//gnum, dennum//gnum gden = numden.gcd(denden) numden, denden = numden//gden, denden//gden return (numnum, numden, dennum, denden) else: return (num, dom.one(), den, dom.one()) def clear_denominators(elts, dom=None): r""" Recursively clear denominators in a list (or other iterable) of elements. Typically intended for elements of fields like QQ(x)(y). """ if not elts: return elts, dom.one() if dom is None: dom = elts[0].parent() if isinstance(dom, FractionField_generic): ring = dom.ring() split = [_clear_denominators_1(elt, ring) for elt in elts] lcmnum = _my_lcm((dennum for _, _, dennum, _ in split)) lcmden = _my_lcm((numden for _, numden, _, _ in split)) num = [(denden*(lcmden//numden))*(numnum*(lcmnum//dennum)) for (numnum, numden, dennum, denden) in split] den = lcmden*lcmnum g = gcd(num + [den]) # XXX: can we be more specific here? num = [p//g for p in num] den = den//g else: num, den = elts, dom.one() # assert all(b/den == a for a, b in zip(elts, num)) # assert gcd(num + [den]).is_one() return num, den -- Marc -- You received this message because you are subscribed to the Google Groups "sage-devel" group. To unsubscribe from this group and stop receiving emails from it, send an email to sage-devel+unsubscr...@googlegroups.com. To post to this group, send email to sage-devel@googlegroups.com. Visit this group at https://groups.google.com/group/sage-devel. For more options, visit https://groups.google.com/d/optout.