Issue #21321: itertools.islice() now releases the reference to the source iterator when the slice is exhausted.
Patch by Anton Afanasyev.
This commit is contained in:
parent
9cc9026294
commit
26f82efe59
|
@ -1,7 +1,7 @@
|
||||||
import unittest
|
import unittest
|
||||||
from test import support
|
from test import support
|
||||||
from itertools import *
|
from itertools import *
|
||||||
from weakref import proxy
|
import weakref
|
||||||
from decimal import Decimal
|
from decimal import Decimal
|
||||||
from fractions import Fraction
|
from fractions import Fraction
|
||||||
import sys
|
import sys
|
||||||
|
@ -1087,6 +1087,15 @@ class TestBasicOps(unittest.TestCase):
|
||||||
list(range(*args)))
|
list(range(*args)))
|
||||||
self.pickletest(islice(range(100), *args))
|
self.pickletest(islice(range(100), *args))
|
||||||
|
|
||||||
|
# Issue #21321: check source iterator is not referenced
|
||||||
|
# from islice() after the latter has been exhausted
|
||||||
|
it = (x for x in (1, 2))
|
||||||
|
wr = weakref.ref(it)
|
||||||
|
it = islice(it, 1)
|
||||||
|
self.assertIsNotNone(wr())
|
||||||
|
list(it) # exhaust the iterator
|
||||||
|
self.assertIsNone(wr())
|
||||||
|
|
||||||
def test_takewhile(self):
|
def test_takewhile(self):
|
||||||
data = [1, 3, 5, 20, 2, 4, 6, 8]
|
data = [1, 3, 5, 20, 2, 4, 6, 8]
|
||||||
self.assertEqual(list(takewhile(underten, data)), [1, 3, 5])
|
self.assertEqual(list(takewhile(underten, data)), [1, 3, 5])
|
||||||
|
@ -1203,7 +1212,7 @@ class TestBasicOps(unittest.TestCase):
|
||||||
|
|
||||||
# test that tee objects are weak referencable
|
# test that tee objects are weak referencable
|
||||||
a, b = tee(range(10))
|
a, b = tee(range(10))
|
||||||
p = proxy(a)
|
p = weakref.proxy(a)
|
||||||
self.assertEqual(getattr(p, '__class__'), type(b))
|
self.assertEqual(getattr(p, '__class__'), type(b))
|
||||||
del a
|
del a
|
||||||
self.assertRaises(ReferenceError, getattr, p, '__class__')
|
self.assertRaises(ReferenceError, getattr, p, '__class__')
|
||||||
|
|
|
@ -17,6 +17,7 @@ Rajiv Abraham
|
||||||
David Abrahams
|
David Abrahams
|
||||||
Marc Abramowitz
|
Marc Abramowitz
|
||||||
Ron Adam
|
Ron Adam
|
||||||
|
Anton Afanasyev
|
||||||
Ali Afshar
|
Ali Afshar
|
||||||
Nitika Agarwal
|
Nitika Agarwal
|
||||||
Jim Ahlstrom
|
Jim Ahlstrom
|
||||||
|
|
|
@ -39,6 +39,9 @@ Core and Builtins
|
||||||
Library
|
Library
|
||||||
-------
|
-------
|
||||||
|
|
||||||
|
- Issue #21321: itertools.islice() now releases the reference to the source
|
||||||
|
iterator when the slice is exhausted. Patch by Anton Afanasyev.
|
||||||
|
|
||||||
- Issue #9815: assertRaises now tries to clear references to local variables
|
- Issue #9815: assertRaises now tries to clear references to local variables
|
||||||
in the exception's traceback.
|
in the exception's traceback.
|
||||||
|
|
||||||
|
|
|
@ -1492,19 +1492,22 @@ islice_next(isliceobject *lz)
|
||||||
Py_ssize_t oldnext;
|
Py_ssize_t oldnext;
|
||||||
PyObject *(*iternext)(PyObject *);
|
PyObject *(*iternext)(PyObject *);
|
||||||
|
|
||||||
|
if (it == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
iternext = *Py_TYPE(it)->tp_iternext;
|
iternext = *Py_TYPE(it)->tp_iternext;
|
||||||
while (lz->cnt < lz->next) {
|
while (lz->cnt < lz->next) {
|
||||||
item = iternext(it);
|
item = iternext(it);
|
||||||
if (item == NULL)
|
if (item == NULL)
|
||||||
return NULL;
|
goto empty;
|
||||||
Py_DECREF(item);
|
Py_DECREF(item);
|
||||||
lz->cnt++;
|
lz->cnt++;
|
||||||
}
|
}
|
||||||
if (stop != -1 && lz->cnt >= stop)
|
if (stop != -1 && lz->cnt >= stop)
|
||||||
return NULL;
|
goto empty;
|
||||||
item = iternext(it);
|
item = iternext(it);
|
||||||
if (item == NULL)
|
if (item == NULL)
|
||||||
return NULL;
|
goto empty;
|
||||||
lz->cnt++;
|
lz->cnt++;
|
||||||
oldnext = lz->next;
|
oldnext = lz->next;
|
||||||
/* The (size_t) cast below avoids the danger of undefined
|
/* The (size_t) cast below avoids the danger of undefined
|
||||||
|
@ -1513,6 +1516,10 @@ islice_next(isliceobject *lz)
|
||||||
if (lz->next < oldnext || (stop != -1 && lz->next > stop))
|
if (lz->next < oldnext || (stop != -1 && lz->next > stop))
|
||||||
lz->next = stop;
|
lz->next = stop;
|
||||||
return item;
|
return item;
|
||||||
|
|
||||||
|
empty:
|
||||||
|
Py_CLEAR(lz->it);
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
|
@ -1522,6 +1529,18 @@ islice_reduce(isliceobject *lz)
|
||||||
* then 'setstate' with the next and count
|
* then 'setstate' with the next and count
|
||||||
*/
|
*/
|
||||||
PyObject *stop;
|
PyObject *stop;
|
||||||
|
if (lz->it == NULL) {
|
||||||
|
PyObject *empty_list;
|
||||||
|
PyObject *empty_it;
|
||||||
|
empty_list = PyList_New(0);
|
||||||
|
if (empty_list == NULL)
|
||||||
|
return NULL;
|
||||||
|
empty_it = PyObject_GetIter(empty_list);
|
||||||
|
Py_DECREF(empty_list);
|
||||||
|
if (empty_it == NULL)
|
||||||
|
return NULL;
|
||||||
|
return Py_BuildValue("O(Nn)n", Py_TYPE(lz), empty_it, 0, 0);
|
||||||
|
}
|
||||||
if (lz->stop == -1) {
|
if (lz->stop == -1) {
|
||||||
stop = Py_None;
|
stop = Py_None;
|
||||||
Py_INCREF(stop);
|
Py_INCREF(stop);
|
||||||
|
|
Loading…
Reference in New Issue