Commit 69b34bfe authored by Raymond Hettinger's avatar Raymond Hettinger

Issue #10323: Predictable final state for slice().

parent ac9a2bb0
...@@ -788,6 +788,11 @@ class TestBasicOps(unittest.TestCase): ...@@ -788,6 +788,11 @@ class TestBasicOps(unittest.TestCase):
self.assertRaises(ValueError, islice, range(10), 1, 'a', 1) self.assertRaises(ValueError, islice, range(10), 1, 'a', 1)
self.assertEqual(len(list(islice(count(), 1, 10, maxsize))), 1) self.assertEqual(len(list(islice(count(), 1, 10, maxsize))), 1)
# Issue #10323: Less islice in a predictable state
c = count()
self.assertEqual(list(islice(c, 1, 3, 50)), [1])
self.assertEqual(next(c), 3)
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]
underten = lambda x: x<10 underten = lambda x: x<10
......
...@@ -43,6 +43,10 @@ Core and Builtins ...@@ -43,6 +43,10 @@ Core and Builtins
Library Library
------- -------
- Issue #10323: itertools.islice() now consumes the minimum number of
inputs before stopping. Formerly, the final state of the underlying
iterator was undefined.
- Issue #10565: The collections.Iterator ABC now checks for both - Issue #10565: The collections.Iterator ABC now checks for both
__iter__ and __next__. __iter__ and __next__.
......
...@@ -1215,6 +1215,7 @@ islice_next(isliceobject *lz) ...@@ -1215,6 +1215,7 @@ islice_next(isliceobject *lz)
{ {
PyObject *item; PyObject *item;
PyObject *it = lz->it; PyObject *it = lz->it;
Py_ssize_t stop = lz->stop;
Py_ssize_t oldnext; Py_ssize_t oldnext;
PyObject *(*iternext)(PyObject *); PyObject *(*iternext)(PyObject *);
...@@ -1226,7 +1227,7 @@ islice_next(isliceobject *lz) ...@@ -1226,7 +1227,7 @@ islice_next(isliceobject *lz)
Py_DECREF(item); Py_DECREF(item);
lz->cnt++; lz->cnt++;
} }
if (lz->stop != -1 && lz->cnt >= lz->stop) if (stop != -1 && lz->cnt >= stop)
return NULL; return NULL;
item = iternext(it); item = iternext(it);
if (item == NULL) if (item == NULL)
...@@ -1234,8 +1235,8 @@ islice_next(isliceobject *lz) ...@@ -1234,8 +1235,8 @@ islice_next(isliceobject *lz)
lz->cnt++; lz->cnt++;
oldnext = lz->next; oldnext = lz->next;
lz->next += lz->step; lz->next += lz->step;
if (lz->next < oldnext) /* Check for overflow */ if (lz->next < oldnext || (stop != -1 && lz->next > stop))
lz->next = lz->stop; lz->next = stop;
return item; return item;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment