diff options
author | Raymond Hettinger <python@rcn.com> | 2004-12-16 10:38:38 (GMT) |
---|---|---|
committer | Raymond Hettinger <python@rcn.com> | 2004-12-16 10:38:38 (GMT) |
commit | 4d01259fb262e7eb035b56ae70637884ef4e8cd8 (patch) | |
tree | 215dc207de652cf3b9b2d6b42b30e3e10c346f5e | |
parent | 8a6a59c58b70df6f3f230ddf575b5af0eb0a2fb6 (diff) | |
download | cpython-4d01259fb262e7eb035b56ae70637884ef4e8cd8.zip cpython-4d01259fb262e7eb035b56ae70637884ef4e8cd8.tar.gz cpython-4d01259fb262e7eb035b56ae70637884ef4e8cd8.tar.bz2 |
SF bug #1085744: Performance issues with PySequence_Tuple()
* Added missing error checks.
* Fixed O(n**2) growth pattern. Modeled after lists to achieve linear
amortized resizing. Improves construction of "tuple(it)" when "it" is
large and does not have a __len__ method. Other cases are unaffected.
-rw-r--r-- | Objects/abstract.c | 18 |
1 files changed, 14 insertions, 4 deletions
diff --git a/Objects/abstract.c b/Objects/abstract.c index 377f359..b981001 100644 --- a/Objects/abstract.c +++ b/Objects/abstract.c @@ -1427,10 +1427,20 @@ PySequence_Tuple(PyObject *v) break; } if (j >= n) { - if (n < 500) - n += 10; - else - n += 100; + int oldn = n; + /* The over-allocation strategy can grow a bit faster + than for lists because unlike lists the + over-allocation isn't permanent -- we reclaim + the excess before the end of this routine. + So, grow by ten and then add 25%. + */ + n += 10; + n += n >> 2; + if (n < oldn) { + /* Check for overflow */ + PyErr_NoMemory(); + goto Fail; + } if (_PyTuple_Resize(&result, n) != 0) { Py_DECREF(item); goto Fail; |