1992-08-13 09:14:11 -03:00
|
|
|
# Example of a generator: re-implement the built-in range function
|
|
|
|
# without actually constructing the list of values. (It turns out
|
|
|
|
# that the built-in function is about 20 times faster -- that's why
|
|
|
|
# it's built-in. :-)
|
|
|
|
|
|
|
|
|
|
|
|
# Wrapper function to emulate the complicated range() arguments
|
|
|
|
|
|
|
|
def range(*a):
|
2003-04-24 14:13:18 -03:00
|
|
|
if len(a) == 1:
|
|
|
|
start, stop, step = 0, a[0], 1
|
|
|
|
elif len(a) == 2:
|
|
|
|
start, stop = a
|
|
|
|
step = 1
|
|
|
|
elif len(a) == 3:
|
|
|
|
start, stop, step = a
|
|
|
|
else:
|
|
|
|
raise TypeError, 'range() needs 1-3 arguments'
|
|
|
|
return Range(start, stop, step)
|
|
|
|
|
1992-08-13 09:14:11 -03:00
|
|
|
|
|
|
|
# Class implementing a range object.
|
|
|
|
# To the user the instances feel like immutable sequences
|
|
|
|
# (and you can't concatenate or slice them)
|
|
|
|
|
|
|
|
class Range:
|
|
|
|
|
2003-04-24 14:13:18 -03:00
|
|
|
# initialization -- should be called only by range() above
|
|
|
|
def __init__(self, start, stop, step):
|
|
|
|
if step == 0:
|
|
|
|
raise ValueError, 'range() called with zero step'
|
|
|
|
self.start = start
|
|
|
|
self.stop = stop
|
|
|
|
self.step = step
|
|
|
|
self.len = max(0, int((self.stop - self.start) / self.step))
|
1992-08-13 09:14:11 -03:00
|
|
|
|
2004-02-12 13:35:32 -04:00
|
|
|
# implement repr(x) and is also used by print x
|
2003-04-24 14:13:18 -03:00
|
|
|
def __repr__(self):
|
2004-02-12 13:35:32 -04:00
|
|
|
return 'range(%r, %r, %r)' % (self.start, self.stop, self.step)
|
1992-08-13 09:14:11 -03:00
|
|
|
|
2003-04-24 14:13:18 -03:00
|
|
|
# implement len(x)
|
|
|
|
def __len__(self):
|
|
|
|
return self.len
|
1992-08-13 09:14:11 -03:00
|
|
|
|
2003-04-24 14:13:18 -03:00
|
|
|
# implement x[i]
|
|
|
|
def __getitem__(self, i):
|
|
|
|
if 0 <= i < self.len:
|
|
|
|
return self.start + self.step * i
|
|
|
|
else:
|
|
|
|
raise IndexError, 'range[i] index out of range'
|
1992-08-13 09:14:11 -03:00
|
|
|
|
|
|
|
|
|
|
|
# Small test program
|
|
|
|
|
|
|
|
def test():
|
2003-04-24 14:13:18 -03:00
|
|
|
import time, __builtin__
|
|
|
|
print range(10), range(-10, 10), range(0, 10, 2)
|
|
|
|
for i in range(100, -100, -10): print i,
|
|
|
|
print
|
|
|
|
t1 = time.time()
|
|
|
|
for i in range(1000):
|
|
|
|
pass
|
|
|
|
t2 = time.time()
|
|
|
|
for i in __builtin__.range(1000):
|
|
|
|
pass
|
|
|
|
t3 = time.time()
|
|
|
|
print t2-t1, 'sec (class)'
|
|
|
|
print t3-t2, 'sec (built-in)'
|
1992-08-13 09:14:11 -03:00
|
|
|
|
|
|
|
|
|
|
|
test()
|