]> granicus.if.org Git - python/commitdiff
Merge 64438: hex/oct/bin can show floats exactly.
authorRaymond Hettinger <python@rcn.com>
Sun, 22 Jun 2008 11:39:13 +0000 (11:39 +0000)
committerRaymond Hettinger <python@rcn.com>
Sun, 22 Jun 2008 11:39:13 +0000 (11:39 +0000)
Include/floatobject.h
Lib/test/test_builtin.py
Objects/abstract.c
Objects/floatobject.c

index 650d5448ac7246256d84db00c710ab4824506d70..ac510dcd34662b802d3ad6f855be753d953f0525 100644 (file)
@@ -111,6 +111,8 @@ PyAPI_FUNC(PyObject *) _PyFloat_FormatAdvanced(PyObject *obj,
                                               Py_UNICODE *format_spec,
                                               Py_ssize_t format_spec_len);
 
+PyAPI_FUNC(PyObject *) _float_to_base(PyObject *v, int base);
+
 #ifdef __cplusplus
 }
 #endif
index 7a898b25de4a4c2450293e74e6e516b10ee35ad7..29c536a2ccd66fec1e4cacae19cbb3ff324ae584 100644 (file)
@@ -553,6 +553,15 @@ class BuiltinTest(unittest.TestCase):
         self.assertEqual(hex(-16), '-0x10')
         self.assertEqual(hex(-16), '-0x10')
         self.assertRaises(TypeError, hex, {})
+        self.assertEqual(hex(3.125), '0x19 * 2.0 ** -3')
+        self.assertEqual(hex(0.0), '0x0 * 2.0 ** 0')
+        for sv in float('nan'), float('inf'), float('-inf'):
+            self.assertEqual(hex(sv), repr(sv))
+        for i in range(100):
+            x = random.expovariate(.05)
+            self.assertEqual(eval(hex(x)), x, (x, hex(x), eval(hex(x))))
+            self.assertEqual(eval(hex(-x)), -x)
+            self.assertEqual(hex(-x), ('-' + hex(x)))
 
     def test_id(self):
         id(None)
@@ -796,6 +805,15 @@ class BuiltinTest(unittest.TestCase):
         self.assertEqual(oct(-100), '-0o144')
         self.assertEqual(oct(-100), '-0o144')
         self.assertRaises(TypeError, oct, ())
+        self.assertEqual(oct(3.125), '0o31 * 2.0 ** -3')
+        self.assertEqual(oct(0.0), '0o0 * 2.0 ** 0')
+        for sv in float('nan'), float('inf'), float('-inf'):
+            self.assertEqual(oct(sv), repr(sv))
+        for i in range(100):
+            x = random.expovariate(.05)
+            self.assertEqual(eval(oct(x)), x)
+            self.assertEqual(eval(oct(-x)), -x)
+            self.assertEqual(oct(-x), ('-' + oct(x)))
 
     def write_testfile(self):
         # NB the first 4 lines are also used to test input, below
@@ -1213,6 +1231,15 @@ class BuiltinTest(unittest.TestCase):
         self.assertEqual(bin(2**65-1), '0b' + '1' * 65)
         self.assertEqual(bin(-(2**65)), '-0b1' + '0' * 65)
         self.assertEqual(bin(-(2**65-1)), '-0b' + '1' * 65)
+        self.assertEqual(bin(3.125), '0b11001 * 2.0 ** -3')
+        self.assertEqual(bin(0.0), '0b0 * 2.0 ** 0')
+        for sv in float('nan'), float('inf'), float('-inf'):
+            self.assertEqual(bin(sv), repr(sv))
+        for i in range(100):
+            x = random.expovariate(.05)
+            self.assertEqual(eval(bin(x)), x)
+            self.assertEqual(eval(bin(-x)), -x)
+            self.assertEqual(bin(-x), ('-' + bin(x)))
 
 class TestSorted(unittest.TestCase):
 
index 60a5e849cfddc8768af29a0d160520df45c7c255..0d40d17d65936c45d8db55f65c6261b224d08b7f 100644 (file)
@@ -1451,8 +1451,11 @@ PyObject *
 PyNumber_ToBase(PyObject *n, int base)
 {
        PyObject *res = NULL;
-       PyObject *index = PyNumber_Index(n);
+       PyObject *index;
 
+       if (PyFloat_Check(n))
+               return _float_to_base(n, base);
+       index = PyNumber_Index(n);
        if (!index)
                return NULL;
        if (PyLong_Check(index))
index db1c99f687ec4c85d0327fc278118a39660eb519..2465fa959a3cfbb482b3551db8b11b588fa076f1 100644 (file)
@@ -1113,6 +1113,36 @@ PyDoc_STRVAR(float_as_integer_ratio_doc,
 ">>> (-.25).as_integer_ratio()\n"
 "(-1, 4)");
 
+PyObject *
+_float_to_base(PyObject *v, int base)
+{
+       PyObject *mant, *conv, *result;
+       double x, fr;
+       int i, exp;
+
+       if (!PyFloat_Check(v)) {
+               PyErr_BadInternalCall();
+               return NULL;
+       }
+       CONVERT_TO_DOUBLE(v, x);
+       if (!Py_IS_FINITE(x))
+               return PyObject_Repr(v);
+       fr = frexp(x, &exp);
+       for (i=0; i<300 && fr != floor(fr) ; i++) {
+               fr *= 2.0;
+               exp--;
+       }
+       mant = PyLong_FromDouble(floor(fr));
+       if (mant == NULL)
+               return NULL;
+       conv = PyNumber_ToBase(mant, base);
+       Py_DECREF(mant);
+       if (conv == NULL)
+               return NULL;
+       result = PyUnicode_FromFormat("%U * 2.0 ** %d", conv, exp);
+       Py_DECREF(conv);
+       return result;
+}
 
 static PyObject *
 float_subtype_new(PyTypeObject *type, PyObject *args, PyObject *kwds);