blob: a0cef742088afa74cbeca526e2f23e051be15916 [file] [log] [blame]
Brett Cannonc2b151c2004-06-27 23:17:35 +00001"""Example of a generator: re-implement the built-in range function
2without actually constructing the list of values.
Guido van Rossume8769491992-08-13 12:14:11 +00003
Brett Cannonc2b151c2004-06-27 23:17:35 +00004OldStyleRange is coded in the way required to work in a 'for' loop before
5iterators were introduced into the language; using __getitem__ and __len__ .
Guido van Rossume8769491992-08-13 12:14:11 +00006
Brett Cannonc2b151c2004-06-27 23:17:35 +00007"""
8def handleargs(arglist):
9 """Take list of arguments and extract/create proper start, stop, and step
10 values and return in a tuple"""
11 try:
12 if len(arglist) == 1:
13 return 0, int(arglist[0]), 1
14 elif len(arglist) == 2:
15 return int(arglist[0]), int(arglist[1]), 1
16 elif len(arglist) == 3:
17 if arglist[2] == 0:
18 raise ValueError("step argument must not be zero")
19 return tuple(int(x) for x in arglist)
20 else:
21 raise TypeError("range() accepts 1-3 arguments, given", len(arglist))
22 except TypeError:
23 raise TypeError("range() arguments must be numbers or strings "
24 "representing numbers")
Guido van Rossume8769491992-08-13 12:14:11 +000025
Brett Cannonc2b151c2004-06-27 23:17:35 +000026def genrange(*a):
27 """Function to implement 'range' as a generator"""
28 start, stop, step = handleargs(a)
29 value = start
30 while value < stop:
31 yield value
32 value += step
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000033
Brett Cannonc2b151c2004-06-27 23:17:35 +000034class oldrange:
35 """Class implementing a range object.
36 To the user the instances feel like immutable sequences
37 (and you can't concatenate or slice them)
Guido van Rossume8769491992-08-13 12:14:11 +000038
Brett Cannonc2b151c2004-06-27 23:17:35 +000039 Done using the old way (pre-iterators; __len__ and __getitem__) to have an
40 object be used by a 'for' loop.
Guido van Rossume8769491992-08-13 12:14:11 +000041
Brett Cannonc2b151c2004-06-27 23:17:35 +000042 """
Guido van Rossume8769491992-08-13 12:14:11 +000043
Brett Cannonc2b151c2004-06-27 23:17:35 +000044 def __init__(self, *a):
45 """ Initialize start, stop, and step values along with calculating the
46 nubmer of values (what __len__ will return) in the range"""
47 self.start, self.stop, self.step = handleargs(a)
48 self.len = max(0, (self.stop - self.start) // self.step)
Guido van Rossume8769491992-08-13 12:14:11 +000049
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000050 def __repr__(self):
Brett Cannonc2b151c2004-06-27 23:17:35 +000051 """implement repr(x) which is also used by print"""
Walter Dörwald70a6b492004-02-12 17:35:32 +000052 return 'range(%r, %r, %r)' % (self.start, self.stop, self.step)
Guido van Rossume8769491992-08-13 12:14:11 +000053
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000054 def __len__(self):
Brett Cannonc2b151c2004-06-27 23:17:35 +000055 """implement len(x)"""
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000056 return self.len
Guido van Rossume8769491992-08-13 12:14:11 +000057
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000058 def __getitem__(self, i):
Brett Cannonc2b151c2004-06-27 23:17:35 +000059 """implement x[i]"""
60 if 0 <= i <= self.len:
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000061 return self.start + self.step * i
62 else:
Collin Winter6f2df4d2007-07-17 20:59:35 +000063 raise IndexError('range[i] index out of range')
Guido van Rossume8769491992-08-13 12:14:11 +000064
65
Guido van Rossume8769491992-08-13 12:14:11 +000066def test():
Georg Brandl1a3284e2007-12-02 09:40:06 +000067 import time, builtins
Brett Cannonc2b151c2004-06-27 23:17:35 +000068 #Just a quick sanity check
Georg Brandl1a3284e2007-12-02 09:40:06 +000069 correct_result = builtins.range(5, 100, 3)
Brett Cannonc2b151c2004-06-27 23:17:35 +000070 oldrange_result = list(oldrange(5, 100, 3))
71 genrange_result = list(genrange(5, 100, 3))
72 if genrange_result != correct_result or oldrange_result != correct_result:
73 raise Exception("error in implementation:\ncorrect = %s"
74 "\nold-style = %s\ngenerator = %s" %
75 (correct_result, oldrange_result, genrange_result))
Collin Winter6f2df4d2007-07-17 20:59:35 +000076 print("Timings for range(1000):")
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000077 t1 = time.time()
Brett Cannonc2b151c2004-06-27 23:17:35 +000078 for i in oldrange(1000):
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000079 pass
80 t2 = time.time()
Brett Cannonc2b151c2004-06-27 23:17:35 +000081 for i in genrange(1000):
Andrew M. Kuchling946c53e2003-04-24 17:13:18 +000082 pass
83 t3 = time.time()
Georg Brandl1a3284e2007-12-02 09:40:06 +000084 for i in builtins.range(1000):
Brett Cannonc2b151c2004-06-27 23:17:35 +000085 pass
86 t4 = time.time()
Collin Winter6f2df4d2007-07-17 20:59:35 +000087 print(t2-t1, 'sec (old-style class)')
88 print(t3-t2, 'sec (generator)')
89 print(t4-t3, 'sec (built-in)')
Guido van Rossume8769491992-08-13 12:14:11 +000090
91
Brett Cannonc2b151c2004-06-27 23:17:35 +000092if __name__ == '__main__':
93 test()