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:
Antoine Pitrou 2014-04-29 12:13:46 +02:00
parent 9cc9026294
commit 26f82efe59
4 changed files with 37 additions and 5 deletions

View File

@ -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__')

View File

@ -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

View File

@ -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.

View File

@ -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);