mirror of
https://github.com/python/cpython.git
synced 2025-07-15 23:35:23 +00:00

svn+ssh://pythondev@svn.python.org/python/trunk ........ r60618 | walter.doerwald | 2008-02-06 15:31:55 +0100 (Wed, 06 Feb 2008) | 6 lines Remove month parameter from Calendar.yeardatescalendar(), Calendar.yeardays2calendar() and Calendar.yeardayscalendar() as the methods don't have such a parameter. Fixes issue #2017. Rewrap content to 80 chars. ........ r60622 | facundo.batista | 2008-02-06 20:28:49 +0100 (Wed, 06 Feb 2008) | 4 lines Fixes issue 1959. Converted tests to unittest. Thanks Giampaolo Rodola. ........ r60626 | thomas.heller | 2008-02-06 21:29:17 +0100 (Wed, 06 Feb 2008) | 3 lines Fixed refcounts and error handling. Should not be merged to py3k branch. ........ r60630 | mark.dickinson | 2008-02-06 23:10:50 +0100 (Wed, 06 Feb 2008) | 4 lines Issue 1979: Make Decimal comparisons (other than !=, ==) involving NaN raise InvalidOperation (and return False if InvalidOperation is trapped). ........ r60632 | mark.dickinson | 2008-02-06 23:25:16 +0100 (Wed, 06 Feb 2008) | 2 lines Remove incorrect usage of :const: in documentation. ........ r60634 | georg.brandl | 2008-02-07 00:45:51 +0100 (Thu, 07 Feb 2008) | 2 lines Revert accidental changes to test_queue in r60605. ........ r60636 | raymond.hettinger | 2008-02-07 01:54:20 +0100 (Thu, 07 Feb 2008) | 1 line Issue 2025: Add tuple.count() and tuple.index() to follow the ABC in collections.Sequence. ........ r60637 | mark.dickinson | 2008-02-07 02:14:23 +0100 (Thu, 07 Feb 2008) | 2 lines Fix broken link in decimal documentation. ........ r60638 | mark.dickinson | 2008-02-07 02:42:06 +0100 (Thu, 07 Feb 2008) | 3 lines IEEE 754 should be IEEE 854; give precise reference for comparisons involving NaNs. ........ r60639 | raymond.hettinger | 2008-02-07 03:12:52 +0100 (Thu, 07 Feb 2008) | 1 line Return ints instead of longs for tuple.count() and tuple.index(). ........ r60640 | raymond.hettinger | 2008-02-07 04:10:33 +0100 (Thu, 07 Feb 2008) | 1 line Merge 60627. ........ r60641 | raymond.hettinger | 2008-02-07 04:25:46 +0100 (Thu, 07 Feb 2008) | 1 line Merge r60628, r60631, and r60633. Register UserList and UserString will the appropriate ABCs. ........ r60642 | brett.cannon | 2008-02-07 08:47:31 +0100 (Thu, 07 Feb 2008) | 3 lines Cast a struct to a void pointer so as to do a type-safe pointer comparison (mistmatch found by clang). ........ r60643 | brett.cannon | 2008-02-07 09:04:07 +0100 (Thu, 07 Feb 2008) | 2 lines Remove unnecessary curly braces around an int literal. ........ r60644 | andrew.kuchling | 2008-02-07 12:43:47 +0100 (Thu, 07 Feb 2008) | 1 line Update URL ........ r60645 | facundo.batista | 2008-02-07 17:16:29 +0100 (Thu, 07 Feb 2008) | 4 lines Fixes issue 2026. Tests converted to unittest. Thanks Giampaolo Rodola. ........ r60646 | christian.heimes | 2008-02-07 18:15:30 +0100 (Thu, 07 Feb 2008) | 1 line Added some statistics code to dict and list object code. I wanted to test how a larger freelist affects the reusage of freed objects. Contrary to my gut feelings 80 objects is more than fine for small apps. I haven't profiled a large app yet. ........ r60648 | facundo.batista | 2008-02-07 20:06:52 +0100 (Thu, 07 Feb 2008) | 6 lines Fixes Issue 1401. When redirected, a possible POST get converted to GET, so it loses its payload. So, it also must lose the headers related to the payload (if it has no content any more, it shouldn't indicate content length and type). ........ r60649 | walter.doerwald | 2008-02-07 20:30:22 +0100 (Thu, 07 Feb 2008) | 3 lines Clarify that the output of TextCalendar.formatmonth() and TextCalendar.formatyear() for custom instances won't be influenced by calls to the module global setfirstweekday() function. Fixes #2018. ........ r60651 | walter.doerwald | 2008-02-07 20:48:34 +0100 (Thu, 07 Feb 2008) | 3 lines Fix documentation for Calendar.iterweekdays(): firstweekday is a property. Fixes second part of #2018. ........ r60653 | walter.doerwald | 2008-02-07 20:57:32 +0100 (Thu, 07 Feb 2008) | 2 lines Fix typo in docstring for Calendar.itermonthdays(). ........ r60655 | raymond.hettinger | 2008-02-07 21:04:37 +0100 (Thu, 07 Feb 2008) | 1 line The float conversion recipe is simpler in Py2.6 ........ r60657 | raymond.hettinger | 2008-02-07 21:10:49 +0100 (Thu, 07 Feb 2008) | 1 line Fix typo ........ r60660 | brett.cannon | 2008-02-07 23:27:10 +0100 (Thu, 07 Feb 2008) | 3 lines Make sure a switch statement does not have repetitive case statements. Error found through LLVM post-2.1 svn. ........ r60661 | christian.heimes | 2008-02-08 01:11:31 +0100 (Fri, 08 Feb 2008) | 1 line Deallocate content of the dict free list on interpreter shutdown ........ r60662 | christian.heimes | 2008-02-08 01:14:34 +0100 (Fri, 08 Feb 2008) | 1 line Use prefix decrement ........ r60663 | amaury.forgeotdarc | 2008-02-08 01:56:02 +0100 (Fri, 08 Feb 2008) | 5 lines issue 2045: Infinite recursion when printing a subclass of defaultdict, if default_factory is set to a bound method. Will backport. ........ r60667 | jeffrey.yasskin | 2008-02-08 07:45:40 +0100 (Fri, 08 Feb 2008) | 2 lines Oops! 2.6's Rational.__ne__ didn't work. ........ r60671 | hyeshik.chang | 2008-02-08 18:10:20 +0100 (Fri, 08 Feb 2008) | 2 lines Update big5hkscs codec to conform to the HKSCS:2004 revision. ........ r60673 | raymond.hettinger | 2008-02-08 23:30:04 +0100 (Fri, 08 Feb 2008) | 4 lines Remove unnecessary modulo division. The preceding test guarantees that 0 <= i < len. ........ r60674 | raymond.hettinger | 2008-02-09 00:02:27 +0100 (Sat, 09 Feb 2008) | 1 line Speed-up __iter__() mixin method. ........ r60675 | raymond.hettinger | 2008-02-09 00:34:21 +0100 (Sat, 09 Feb 2008) | 1 line Fill-in missing Set comparisons ........ r60677 | raymond.hettinger | 2008-02-09 00:57:06 +0100 (Sat, 09 Feb 2008) | 1 line Add advice on choosing between DictMixin and MutableMapping ........
217 lines
6.4 KiB
Python
217 lines
6.4 KiB
Python
""" Test Iterator Length Transparency
|
|
|
|
Some functions or methods which accept general iterable arguments have
|
|
optional, more efficient code paths if they know how many items to expect.
|
|
For instance, map(func, iterable), will pre-allocate the exact amount of
|
|
space required whenever the iterable can report its length.
|
|
|
|
The desired invariant is: len(it)==len(list(it)).
|
|
|
|
A complication is that an iterable and iterator can be the same object. To
|
|
maintain the invariant, an iterator needs to dynamically update its length.
|
|
For instance, an iterable such as range(10) always reports its length as ten,
|
|
but it=iter(range(10)) starts at ten, and then goes to nine after next(it).
|
|
Having this capability means that map() can ignore the distinction between
|
|
map(func, iterable) and map(func, iter(iterable)).
|
|
|
|
When the iterable is immutable, the implementation can straight-forwardly
|
|
report the original length minus the cumulative number of calls to next().
|
|
This is the case for tuples, range objects, and itertools.repeat().
|
|
|
|
Some containers become temporarily immutable during iteration. This includes
|
|
dicts, sets, and collections.deque. Their implementation is equally simple
|
|
though they need to permantently set their length to zero whenever there is
|
|
an attempt to iterate after a length mutation.
|
|
|
|
The situation slightly more involved whenever an object allows length mutation
|
|
during iteration. Lists and sequence iterators are dynanamically updatable.
|
|
So, if a list is extended during iteration, the iterator will continue through
|
|
the new items. If it shrinks to a point before the most recent iteration,
|
|
then no further items are available and the length is reported at zero.
|
|
|
|
Reversed objects can also be wrapped around mutable objects; however, any
|
|
appends after the current position are ignored. Any other approach leads
|
|
to confusion and possibly returning the same item more than once.
|
|
|
|
The iterators not listed above, such as enumerate and the other itertools,
|
|
are not length transparent because they have no way to distinguish between
|
|
iterables that report static length and iterators whose length changes with
|
|
each call (i.e. the difference between enumerate('abc') and
|
|
enumerate(iter('abc')).
|
|
|
|
"""
|
|
|
|
import unittest
|
|
from test import test_support
|
|
from itertools import repeat
|
|
from collections import deque
|
|
from builtins import len as _len
|
|
|
|
n = 10
|
|
|
|
def len(obj):
|
|
try:
|
|
return _len(obj)
|
|
except TypeError:
|
|
try:
|
|
# note: this is an internal undocumented API,
|
|
# don't rely on it in your own programs
|
|
return obj.__length_hint__()
|
|
except AttributeError:
|
|
raise TypeError
|
|
|
|
class TestInvariantWithoutMutations(unittest.TestCase):
|
|
|
|
def test_invariant(self):
|
|
it = self.it
|
|
for i in reversed(range(1, n+1)):
|
|
self.assertEqual(len(it), i)
|
|
next(it)
|
|
self.assertEqual(len(it), 0)
|
|
self.assertRaises(StopIteration, next, it)
|
|
self.assertEqual(len(it), 0)
|
|
|
|
class TestTemporarilyImmutable(TestInvariantWithoutMutations):
|
|
|
|
def test_immutable_during_iteration(self):
|
|
# objects such as deques, sets, and dictionaries enforce
|
|
# length immutability during iteration
|
|
|
|
it = self.it
|
|
self.assertEqual(len(it), n)
|
|
next(it)
|
|
self.assertEqual(len(it), n-1)
|
|
self.mutate()
|
|
self.assertRaises(RuntimeError, next, it)
|
|
self.assertEqual(len(it), 0)
|
|
|
|
## ------- Concrete Type Tests -------
|
|
|
|
class TestRepeat(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = repeat(None, n)
|
|
|
|
def test_no_len_for_infinite_repeat(self):
|
|
# The repeat() object can also be infinite
|
|
self.assertRaises(TypeError, len, repeat(None))
|
|
|
|
class TestXrange(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = iter(range(n))
|
|
|
|
class TestXrangeCustomReversed(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = reversed(range(n))
|
|
|
|
class TestTuple(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = iter(tuple(range(n)))
|
|
|
|
## ------- Types that should not be mutated during iteration -------
|
|
|
|
class TestDeque(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = deque(range(n))
|
|
self.it = iter(d)
|
|
self.mutate = d.pop
|
|
|
|
class TestDequeReversed(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = deque(range(n))
|
|
self.it = reversed(d)
|
|
self.mutate = d.pop
|
|
|
|
class TestDictKeys(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = dict.fromkeys(range(n))
|
|
self.it = iter(d)
|
|
self.mutate = d.popitem
|
|
|
|
class TestDictItems(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = dict.fromkeys(range(n))
|
|
self.it = iter(d.items())
|
|
self.mutate = d.popitem
|
|
|
|
class TestDictValues(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = dict.fromkeys(range(n))
|
|
self.it = iter(d.values())
|
|
self.mutate = d.popitem
|
|
|
|
class TestSet(TestTemporarilyImmutable):
|
|
|
|
def setUp(self):
|
|
d = set(range(n))
|
|
self.it = iter(d)
|
|
self.mutate = d.pop
|
|
|
|
## ------- Types that can mutate during iteration -------
|
|
|
|
class TestList(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = iter(range(n))
|
|
|
|
def test_mutation(self):
|
|
d = list(range(n))
|
|
it = iter(d)
|
|
next(it)
|
|
next(it)
|
|
self.assertEqual(len(it), n-2)
|
|
d.append(n)
|
|
self.assertEqual(len(it), n-1) # grow with append
|
|
d[1:] = []
|
|
self.assertEqual(len(it), 0)
|
|
self.assertEqual(list(it), [])
|
|
d.extend(range(20))
|
|
self.assertEqual(len(it), 0)
|
|
|
|
class TestListReversed(TestInvariantWithoutMutations):
|
|
|
|
def setUp(self):
|
|
self.it = reversed(range(n))
|
|
|
|
def test_mutation(self):
|
|
d = list(range(n))
|
|
it = reversed(d)
|
|
next(it)
|
|
next(it)
|
|
self.assertEqual(len(it), n-2)
|
|
d.append(n)
|
|
self.assertEqual(len(it), n-2) # ignore append
|
|
d[1:] = []
|
|
self.assertEqual(len(it), 0)
|
|
self.assertEqual(list(it), []) # confirm invariant
|
|
d.extend(range(20))
|
|
self.assertEqual(len(it), 0)
|
|
|
|
|
|
def test_main():
|
|
unittests = [
|
|
TestRepeat,
|
|
TestXrange,
|
|
TestXrangeCustomReversed,
|
|
TestTuple,
|
|
TestDeque,
|
|
TestDequeReversed,
|
|
TestDictKeys,
|
|
TestDictItems,
|
|
TestDictValues,
|
|
TestSet,
|
|
TestList,
|
|
TestListReversed,
|
|
]
|
|
test_support.run_unittest(*unittests)
|
|
|
|
if __name__ == "__main__":
|
|
test_main()
|