-í
é¶<c      s     d  k  l Z  d Z Š d Z /d Z ëd Z ‹d Z @d „  Z Ud „  Z ’d „  Z ¶d	 f  d
 „  ƒ  YZ	 íd f  d „  ƒ  YZ
 ¬d Z ?h  e d <e d <e d <e d <e d <e d <Z Je d „ Z Ue d j o Ve d ƒ n d S(   (   s
   generatorssÜ	  
Let's try a simple generator:

    >>> def f():
    ...    yield 1
    ...    yield 2

    >>> for i in f():
    ...     print i
    1
    2
    >>> g = f()
    >>> g.next()
    1
    >>> g.next()
    2

"Falling off the end" stops the generator:

    >>> g.next()
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
      File "<stdin>", line 2, in g
    StopIteration

"return" also stops the generator:

    >>> def f():
    ...     yield 1
    ...     return
    ...     yield 2 # never reached
    ...
    >>> g = f()
    >>> g.next()
    1
    >>> g.next()
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
      File "<stdin>", line 3, in f
    StopIteration
    >>> g.next() # once stopped, can't be resumed
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
    StopIteration

"raise StopIteration" stops the generator too:

    >>> def f():
    ...     yield 1
    ...     raise StopIteration
    ...     yield 2 # never reached
    ...
    >>> g = f()
    >>> g.next()
    1
    >>> g.next()
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
    StopIteration
    >>> g.next()
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
    StopIteration

However, they are not exactly equivalent:

    >>> def g1():
    ...     try:
    ...         return
    ...     except:
    ...         yield 1
    ...
    >>> list(g1())
    []

    >>> def g2():
    ...     try:
    ...         raise StopIteration
    ...     except:
    ...         yield 42
    >>> print list(g2())
    [42]

This may be surprising at first:

    >>> def g3():
    ...     try:
    ...         return
    ...     finally:
    ...         yield 1
    ...
    >>> list(g3())
    [1]

Let's create an alternate range() function implemented as a generator:

    >>> def yrange(n):
    ...     for i in range(n):
    ...         yield i
    ...
    >>> list(yrange(5))
    [0, 1, 2, 3, 4]

Generators always return to the most recent caller:

    >>> def creator():
    ...     r = yrange(5)
    ...     print "creator", r.next()
    ...     return r
    ...
    >>> def caller():
    ...     r = creator()
    ...     for i in r:
    ...             print "caller", i
    ...
    >>> caller()
    creator 0
    caller 1
    caller 2
    caller 3
    caller 4

Generators can call other generators:

    >>> def zrange(n):
    ...     for i in yrange(n):
    ...         yield i
    ...
    >>> list(zrange(5))
    [0, 1, 2, 3, 4]

sm  

Specification:  Yield

    Restriction:  A generator cannot be resumed while it is actively
    running:

    >>> def g():
    ...     i = me.next()
    ...     yield i
    >>> me = g()
    >>> me.next()
    Traceback (most recent call last):
     ...
      File "<string>", line 2, in g
    ValueError: generator already executing

Specification: Return

    Note that return isn't always equivalent to raising StopIteration:  the
    difference lies in how enclosing try/except constructs are treated.
    For example,

        >>> def f1():
        ...     try:
        ...         return
        ...     except:
        ...        yield 1
        >>> print list(f1())
        []

    because, as in any function, return simply exits, but

        >>> def f2():
        ...     try:
        ...         raise StopIteration
        ...     except:
        ...         yield 42
        >>> print list(f2())
        [42]

    because StopIteration is captured by a bare "except", as is any
    exception.

Specification: Generators and Exception Propagation

    >>> def f():
    ...     return 1//0
    >>> def g():
    ...     yield f()  # the zero division exception propagates
    ...     yield 42   # and we'll never get here
    >>> k = g()
    >>> k.next()
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
      File "<stdin>", line 2, in g
      File "<stdin>", line 2, in f
    ZeroDivisionError: integer division or modulo by zero
    >>> k.next()  # and the generator cannot be resumed
    Traceback (most recent call last):
      File "<stdin>", line 1, in ?
    StopIteration
    >>>

Specification: Try/Except/Finally

    >>> def f():
    ...     try:
    ...         yield 1
    ...         try:
    ...             yield 2
    ...             1//0
    ...             yield 3  # never get here
    ...         except ZeroDivisionError:
    ...             yield 4
    ...             yield 5
    ...             raise
    ...         except:
    ...             yield 6
    ...         yield 7     # the "raise" above stops this
    ...     except:
    ...         yield 8
    ...     yield 9
    ...     try:
    ...         x = 12
    ...     finally:
    ...         yield 10
    ...     yield 11
    >>> print list(f())
    [1, 2, 4, 5, 8, 9, 10, 11]
    >>>

Guido's binary tree example.

    >>> # A binary tree class.
    >>> class Tree:
    ...
    ...     def __init__(self, label, left=None, right=None):
    ...         self.label = label
    ...         self.left = left
    ...         self.right = right
    ...
    ...     def __repr__(self, level=0, indent="    "):
    ...         s = level*indent + `self.label`
    ...         if self.left:
    ...             s = s + "\n" + self.left.__repr__(level+1, indent)
    ...         if self.right:
    ...             s = s + "\n" + self.right.__repr__(level+1, indent)
    ...         return s
    ...
    ...     def __iter__(self):
    ...         return inorder(self)

    >>> # Create a Tree from a list.
    >>> def tree(list):
    ...     n = len(list)
    ...     if n == 0:
    ...         return []
    ...     i = n // 2
    ...     return Tree(list[i], tree(list[:i]), tree(list[i+1:]))

    >>> # Show it off: create a tree.
    >>> t = tree("ABCDEFGHIJKLMNOPQRSTUVWXYZ")

    >>> # A recursive generator that generates Tree leaves in in-order.
    >>> def inorder(t):
    ...     if t:
    ...         for x in inorder(t.left):
    ...             yield x
    ...         yield t.label
    ...         for x in inorder(t.right):
    ...             yield x

    >>> # Show it off: create a tree.
    ... t = tree("ABCDEFGHIJKLMNOPQRSTUVWXYZ")
    ... # Print the nodes of the tree in in-order.
    ... for x in t:
    ...     print x,
    A B C D E F G H I J K L M N O P Q R S T U V W X Y Z

    >>> # A non-recursive generator.
    >>> def inorder(node):
    ...     stack = []
    ...     while node:
    ...         while node.left:
    ...             stack.append(node)
    ...             node = node.left
    ...         yield node.label
    ...         while not node.right:
    ...             try:
    ...                 node = stack.pop()
    ...             except IndexError:
    ...                 return
    ...             yield node.label
    ...         node = node.right

    >>> # Exercise the non-recursive generator.
    >>> for x in t:
    ...     print x,
    A B C D E F G H I J K L M N O P Q R S T U V W X Y Z

s¥  

The difference between yielding None and returning it.

>>> def g():
...     for i in range(3):
...         yield None
...     yield None
...     return
>>> list(g())
[None, None, None, None]

Ensure that explicitly raising StopIteration acts like any other exception
in try/except, not like a return.

>>> def g():
...     yield 1
...     try:
...         raise StopIteration
...     except:
...         yield 2
...     yield 3
>>> list(g())
[1, 2, 3]

Next one was posted to c.l.py.

>>> def gcomb(x, k):
...     "Generate all combinations of k elements from list x."
...
...     if k > len(x):
...         return
...     if k == 0:
...         yield []
...     else:
...         first, rest = x[0], x[1:]
...         # A combination does or doesn't contain first.
...         # If it does, the remainder is a k-1 comb of rest.
...         for c in gcomb(rest, k-1):
...             c.insert(0, first)
...             yield c
...         # If it doesn't contain first, it's a k comb of rest.
...         for c in gcomb(rest, k):
...             yield c

>>> seq = range(1, 5)
>>> for k in range(len(seq) + 2):
...     print "%d-combs of %s:" % (k, seq)
...     for c in gcomb(seq, k):
...         print "   ", c
0-combs of [1, 2, 3, 4]:
    []
1-combs of [1, 2, 3, 4]:
    [1]
    [2]
    [3]
    [4]
2-combs of [1, 2, 3, 4]:
    [1, 2]
    [1, 3]
    [1, 4]
    [2, 3]
    [2, 4]
    [3, 4]
3-combs of [1, 2, 3, 4]:
    [1, 2, 3]
    [1, 2, 4]
    [1, 3, 4]
    [2, 3, 4]
4-combs of [1, 2, 3, 4]:
    [1, 2, 3, 4]
5-combs of [1, 2, 3, 4]:

From the Iterators list, about the types of these things.

>>> def g():
...     yield 1
...
>>> type(g)
<type 'function'>
>>> i = g()
>>> type(i)
<type 'generator'>
>>> [s for s in dir(i) if not s.startswith('_')]
['gi_frame', 'gi_running', 'next']
>>> print i.next.__doc__
x.next() -> the next value, or raise StopIteration
>>> iter(i) is i
1
>>> import types
>>> isinstance(i, types.GeneratorType)
1

And more, added later.

>>> i.gi_running
0
>>> type(i.gi_frame)
<type 'frame'>
>>> i.gi_running = 42
Traceback (most recent call last):
  ...
TypeError: readonly attribute
>>> def g():
...     yield me.gi_running
>>> me = g()
>>> me.gi_running
0
>>> me.next()
1
>>> me.gi_running
0

A clever union-find implementation from c.l.py, due to David Eppstein.
Sent: Friday, June 29, 2001 12:16 PM
To: python-list@python.org
Subject: Re: PEP 255: Simple Generators

>>> class disjointSet:
...     def __init__(self, name):
...         self.name = name
...         self.parent = None
...         self.generator = self.generate()
...
...     def generate(self):
...         while not self.parent:
...             yield self
...         for x in self.parent.generator:
...             yield x
...
...     def find(self):
...         return self.generator.next()
...
...     def union(self, parent):
...         if self.parent:
...             raise ValueError("Sorry, I'm not a root!")
...         self.parent = parent
...
...     def __str__(self):
...         return self.name

>>> names = "ABCDEFGHIJKLM"
>>> sets = [disjointSet(name) for name in names]
>>> roots = sets[:]

>>> import random
>>> random.seed(42)
>>> while 1:
...     for s in sets:
...         print "%s->%s" % (s, s.find()),
...     print
...     if len(roots) > 1:
...         s1 = random.choice(roots)
...         roots.remove(s1)
...         s2 = random.choice(roots)
...         s1.union(s2)
...         print "merged", s1, "into", s2
...     else:
...         break
A->A B->B C->C D->D E->E F->F G->G H->H I->I J->J K->K L->L M->M
merged D into G
A->A B->B C->C D->G E->E F->F G->G H->H I->I J->J K->K L->L M->M
merged C into F
A->A B->B C->F D->G E->E F->F G->G H->H I->I J->J K->K L->L M->M
merged L into A
A->A B->B C->F D->G E->E F->F G->G H->H I->I J->J K->K L->A M->M
merged H into E
A->A B->B C->F D->G E->E F->F G->G H->E I->I J->J K->K L->A M->M
merged B into E
A->A B->E C->F D->G E->E F->F G->G H->E I->I J->J K->K L->A M->M
merged J into G
A->A B->E C->F D->G E->E F->F G->G H->E I->I J->G K->K L->A M->M
merged E into G
A->A B->G C->F D->G E->G F->F G->G H->G I->I J->G K->K L->A M->M
merged M into G
A->A B->G C->F D->G E->G F->F G->G H->G I->I J->G K->K L->A M->G
merged I into K
A->A B->G C->F D->G E->G F->F G->G H->G I->K J->G K->K L->A M->G
merged K into A
A->A B->G C->F D->G E->G F->F G->G H->G I->A J->G K->A L->A M->G
merged F into A
A->A B->G C->A D->G E->G F->A G->G H->G I->A J->G K->A L->A M->G
merged A into G
A->G B->G C->G D->G E->G F->G G->G H->G I->G J->G K->G L->G M->G
s/  

Build up to a recursive Sieve of Eratosthenes generator.

>>> def firstn(g, n):
...     return [g.next() for i in range(n)]

>>> def intsfrom(i):
...     while 1:
...         yield i
...         i += 1

>>> firstn(intsfrom(5), 7)
[5, 6, 7, 8, 9, 10, 11]

>>> def exclude_multiples(n, ints):
...     for i in ints:
...         if i % n:
...             yield i

>>> firstn(exclude_multiples(3, intsfrom(1)), 6)
[1, 2, 4, 5, 7, 8]

>>> def sieve(ints):
...     prime = ints.next()
...     yield prime
...     not_divisible_by_prime = exclude_multiples(prime, ints)
...     for p in sieve(not_divisible_by_prime):
...         yield p

>>> primes = sieve(intsfrom(2))
>>> firstn(primes, 20)
[2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71]


Another famous problem:  generate all integers of the form
    2**i * 3**j  * 5**k
in increasing order, where i,j,k >= 0.  Trickier than it may look at first!
Try writing it without generators, and correctly, and without generating
3 internal results for each result output.

>>> def times(n, g):
...     for i in g:
...         yield n * i
>>> firstn(times(10, intsfrom(1)), 10)
[10, 20, 30, 40, 50, 60, 70, 80, 90, 100]

>>> def merge(g, h):
...     ng = g.next()
...     nh = h.next()
...     while 1:
...         if ng < nh:
...             yield ng
...             ng = g.next()
...         elif ng > nh:
...             yield nh
...             nh = h.next()
...         else:
...             yield ng
...             ng = g.next()
...             nh = h.next()

The following works, but is doing a whale of a lot of redundant work --
it's not clear how to get the internal uses of m235 to share a single
generator.  Note that me_times2 (etc) each need to see every element in the
result sequence.  So this is an example where lazy lists are more natural
(you can look at the head of a lazy list any number of times).

>>> def m235():
...     yield 1
...     me_times2 = times(2, m235())
...     me_times3 = times(3, m235())
...     me_times5 = times(5, m235())
...     for i in merge(merge(me_times2,
...                          me_times3),
...                    me_times5):
...         yield i

Don't print "too many" of these -- the implementation above is extremely
inefficient:  each call of m235() leads to 3 recursive calls, and in
turn each of those 3 more, and so on, and so on, until we've descended
enough levels to satisfy the print stmts.  Very odd:  when I printed 5
lines of results below, this managed to screw up Win98's malloc in "the
usual" way, i.e. the heap grew over 4Mb so Win98 started fragmenting
address space, and it *looked* like a very slow leak.

>>> result = m235()
>>> for i in range(3):
...     print firstn(result, 15)
[1, 2, 3, 4, 5, 6, 8, 9, 10, 12, 15, 16, 18, 20, 24]
[25, 27, 30, 32, 36, 40, 45, 48, 50, 54, 60, 64, 72, 75, 80]
[81, 90, 96, 100, 108, 120, 125, 128, 135, 144, 150, 160, 162, 180, 192]

Heh.  Here's one way to get a shared list, complete with an excruciating
namespace renaming trick.  The *pretty* part is that the times() and merge()
functions can be reused as-is, because they only assume their stream
arguments are iterable -- a LazyList is the same as a generator to times().

>>> class LazyList:
...     def __init__(self, g):
...         self.sofar = []
...         self.fetch = g.next
...
...     def __getitem__(self, i):
...         sofar, fetch = self.sofar, self.fetch
...         while i >= len(sofar):
...             sofar.append(fetch())
...         return sofar[i]

>>> def m235():
...     yield 1
...     # Gack:  m235 below actually refers to a LazyList.
...     me_times2 = times(2, m235)
...     me_times3 = times(3, m235)
...     me_times5 = times(5, m235)
...     for i in merge(merge(me_times2,
...                          me_times3),
...                    me_times5):
...         yield i

Print as many of these as you like -- *this* implementation is memory-
efficient.

>>> m235 = LazyList(m235())
>>> for i in range(5):
...     print [m235[j] for j in range(15*i, 15*(i+1))]
[1, 2, 3, 4, 5, 6, 8, 9, 10, 12, 15, 16, 18, 20, 24]
[25, 27, 30, 32, 36, 40, 45, 48, 50, 54, 60, 64, 72, 75, 80]
[81, 90, 96, 100, 108, 120, 125, 128, 135, 144, 150, 160, 162, 180, 192]
[200, 216, 225, 240, 243, 250, 256, 270, 288, 300, 320, 324, 360, 375, 384]
[400, 405, 432, 450, 480, 486, 500, 512, 540, 576, 600, 625, 640, 648, 675]


Ye olde Fibonacci generator, LazyList style.

>>> def fibgen(a, b):
...
...     def sum(g, h):
...         while 1:
...             yield g.next() + h.next()
...
...     def tail(g):
...         g.next()    # throw first away
...         for x in g:
...             yield x
...
...     yield a
...     yield b
...     for s in sum(iter(fib),
...                  tail(iter(fib))):
...         yield s

>>> fib = LazyList(fibgen(1, 2))
>>> firstn(iter(fib), 17)
[1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, 610, 987, 1597, 2584]
s  

>>> def f():
...     return 22
...     yield 1
Traceback (most recent call last):
  ...
SyntaxError: 'return' with argument inside generator (<string>, line 2)

>>> def f():
...     yield 1
...     return 22
Traceback (most recent call last):
  ...
SyntaxError: 'return' with argument inside generator (<string>, line 3)

"return None" is not the same as "return" in a generator:

>>> def f():
...     yield 1
...     return None
Traceback (most recent call last):
  ...
SyntaxError: 'return' with argument inside generator (<string>, line 3)

This one is fine:

>>> def f():
...     yield 1
...     return

>>> def f():
...     try:
...         yield 1
...     finally:
...         pass
Traceback (most recent call last):
  ...
SyntaxError: 'yield' not allowed in a 'try' block with a 'finally' clause (<string>, line 3)

>>> def f():
...     try:
...         try:
...             1//0
...         except ZeroDivisionError:
...             yield 666  # bad because *outer* try has finally
...         except:
...             pass
...     finally:
...         pass
Traceback (most recent call last):
  ...
SyntaxError: 'yield' not allowed in a 'try' block with a 'finally' clause (<string>, line 6)

But this is fine:

>>> def f():
...     try:
...         try:
...             yield 12
...             1//0
...         except ZeroDivisionError:
...             yield 666
...         except:
...             try:
...                 x = 12
...             finally:
...                 yield 12
...     except:
...         return
>>> list(f())
[12, 666]

>>> def f():
...    yield
Traceback (most recent call last):
SyntaxError: invalid syntax

>>> def f():
...    if 0:
...        yield
Traceback (most recent call last):
SyntaxError: invalid syntax

>>> def f():
...     if 0:
...         yield 1
>>> type(f())
<type 'generator'>

>>> def f():
...    if "":
...        yield None
>>> type(f())
<type 'generator'>

>>> def f():
...     return
...     try:
...         if x==4:
...             pass
...         elif 0:
...             try:
...                 1//0
...             except SyntaxError:
...                 pass
...             else:
...                 if 0:
...                     while 12:
...                         x += 1
...                         yield 2 # don't blink
...                         f(a, b, c, d, e)
...         else:
...             pass
...     except:
...         x = 1
...     return
>>> type(f())
<type 'generator'>

>>> def f():
...     if 0:
...         def g():
...             yield 1
...
>>> type(f())
<type 'NoneType'>

>>> def f():
...     if 0:
...         class C:
...             def __init__(self):
...                 yield 1
...             def f(self):
...                 yield 2
>>> type(f())
<type 'NoneType'>

>>> def f():
...     if 0:
...         return
...     if 0:
...         yield 2
>>> type(f())
<type 'generator'>


>>> def f():
...     if 0:
...         lambda x:  x        # shouldn't trigger here
...         return              # or here
...         def f(i):
...             return 2*i      # or here
...         if 0:
...             return 3        # but *this* sucks (line 8)
...     if 0:
...         yield 2             # because it's a generator
Traceback (most recent call last):
SyntaxError: 'return' with argument inside generator (<string>, line 8)
c   #sV   @Bt  g t ˆ  ƒ } D| ‡  ‡ d † ‰ Lx ˆ d ƒ DL] } M| Vq> Wd  S(   Nc   #sr   DE|  t ˆ  ƒ j o F| VnK HxD ˆ  |  ƒ  DH]2 | |  <Ix" ˆ |  d ƒ DI] } J| VqV Wq5 Wd  S(   Ni   (   s   is   lens   gss   valuess   gens   x(   s   is   valuess   x(   s   gss   gen(    s*   /usr/lib/python2.2/test/test_generators.pys   genDs     	i    (   s   Nones   lens   gss   valuess   gens   x(   s   gss   valuess   xs   gen(    (   s   gss   gens*   /usr/lib/python2.2/test/test_generators.pys   conjoin@s
    	c  	 #s}   UWt  ˆ  ƒ ‰ Xt g ˆ } ]| ‡  ‡ ‡ ‡ d † ‰ o| ‡  ‡ ‡ d † ‰ „x ˆ d ƒ D„] } …| Vqe Wd  S(   Nc   #s¯   ]^|  ˆ j o _| VnŽ aˆ |  d oW b|  d } cx@ ˆ  |  ƒ  Dc]. | |  <dx ˆ | ƒ Dd] } e| Vqk WqN Wn% hx ˆ |  ƒ Dh] } i| Vq— Wd  S(   Ni   i   (   s   is   ns   valuess   ip1s   gss   gens   xs   _gen3(   s   is   valuess   ip1s   x(   s   gss   _gen3s   gens   n(    s*   /usr/lib/python2.2/test/test_generators.pys   gen]s     	 	c 	  #sm  opt  o) |  ˆ j  o ˆ |  d d j p t ‚ q|  d |  d |  d f \ } } } rˆ  |  | !\ } } } t| ˆ j oe vx[ | ƒ  Dv]M | |  <wx= | ƒ  Dw]/ | | <xx | ƒ  Dx] | | <y| VqÇ Wq­ Wq“ Wn ~xx | ƒ  D~]j | |  <xZ | ƒ  D]L | | <€x< | ƒ  D€]. | | <x ˆ | ƒ D] } ‚| VqIWq,WqWqø Wd  S(   Ni   i    i   i   (   s	   __debug__s   is   ns   AssertionErrors   ip1s   ip2s   ip3s   gss   gs   g1s   g2s   valuess   _gen3s   x(	   s   is   valuess   gs   g1s   ip2s   ip3s   ip1s   xs   g2(   s   gss   ns   _gen3(    s*   /usr/lib/python2.2/test/test_generators.pys   _gen3os(   3'       	i    (   s   lens   gss   ns   Nones   valuess   gens   _gen3s   x(   s   gss   valuess   xs   _gen3s   gens   n(    (   s   gss   _gen3s   gens   ns*   /usr/lib/python2.2/test/test_generators.pys   conjoinUs    	c   #s”  ’“t  |  ƒ } ”t g | } •t g | } –t } —d } ˜xF˜d o;šyW ›xM ›| | j  o< œ|  | ƒ  i	 } | | <| ƒ  | | <ž| d 7} q` WWn Ÿ| j
 o
  n' X¢t o | | j p t ‚ £| V¦| d 8} §x‰ §| d j oV ¨y) ©| | ƒ  | | <«| d 7} ¬PWn# ­| j
 o ¯| d 8} n XqW±t o | d j  p t ‚ ²PqJ Wd  S(   Ni    i   (   s   lens   gss   ns   Nones   valuess   iterss   StopIterations   _StopIterations   is   nexts   its	   __debug__s   AssertionError(   s   gss   is   valuess   iterss   its   _StopIterations   n(    (    s*   /usr/lib/python2.2/test/test_generators.pys   flat_conjoin’s8   		 
  s   Queensc     s/   ¶t  Z ·d „  Z Ød „  Z Ýd „  Z RS(   Nc  	 sÒ   ·¸| ˆ  _  ¹t | ƒ ‰ Æg  ˆ  _ Çxž ˆ DÇ]“ } Èg  i } ˆ DË]H } | d | >d | | | | d >Bd | d | d | | >Bƒ qN ~ } Í| ‡  ‡ d † } Õˆ  i i | ƒ q4 Wd  S(   Nl    i   i   c   #sr   ÍÎxe ˆ DÎ]Z } Ï|  | } Ð| ˆ  i @d j o0 Ñˆ  i | O_ Ò| VÓˆ  i | M_ n q Wd  S(   Ni    (   s   rangens   js   rowusess   usess   selfs   used(   s   rowusess   usess   j(   s   selfs   rangen(    s*   /usr/lib/python2.2/test/test_generators.pys   rowgenÍs   
 	(   s   ns   selfs   ranges   rangens   rowgeneratorss   is   appends   _[1]s   js   rowusess   rowgen(   s   selfs   ns   _[1]s   js   is   rowusess   rowgens   rangen(    (   s   selfs   rangens*   /usr/lib/python2.2/test/test_generators.pys   __init__·s   
 	Tc   #s:   ØÙd |  _ Úx! t |  i ƒ DÚ] } Û| Vq" Wd  S(   Ni    (   s   selfs   useds   conjoins   rowgeneratorss   row2col(   s   selfs   row2col(    (    s*   /usr/lib/python2.2/test/test_generators.pys   solveØs    	c   sÒ   ÝÞ|  i } ßt o | t | ƒ j p t ‚ àd d | } á| GHâx| t | ƒ Dâ]k } ãg  i	 } t | ƒ Dã] } | d ƒ q| ~ } äd | | | <åd d i | ƒ d GHæ| GHq\ Wd  S(   Ns   +s   -+s    s   Qs   |(   s   selfs   ns	   __debug__s   lens   row2cols   AssertionErrors   seps   ranges   is   appends   _[1]s   js   squaress   join(   s   selfs   row2cols   _[1]s   squaress   js   seps   is   n(    (    s*   /usr/lib/python2.2/test/test_generators.pys   printsolutionÝs   $ 	 (   s   __name__s
   __module__s   __init__s   solves   printsolution(    (    (    s*   /usr/lib/python2.2/test/test_generators.pys   Queens¶s   	!s   Knightsc     sV   ít  Z îd d „ Z yd „  Z ~d „  Z ‚d „  Z “d „  Z ˜d „  Z RS(   Ni    c   sK  îïˆ ˆ f \ ˆ  _  ˆ  _ óg  ‰ ˆ  _ ùt ‡ d † ‰ ‡ d †  ‰ ‡ ‡  ‡ ‡ ‡ d †  } !‡  ‡ ‡ ‡ ‡ ‡ d †  } :t ‡ ‡ ‡  ‡ d † } Tˆ d d ˆ d d t ‡  ‡ ‡ ‡ d † }
 n‡  ‡ d	 †  } rˆ ˆ d
 j  o s| g ˆ  _ n; u| | g | o |
 p | g ˆ ˆ d | g ˆ  _ d  S(   Nc   s·   ùd } } xƒ ˆ  |  D]t } ˆ  | } | i |  ƒ | | ƒ } | d j o | d 7} n" 	| d j o 
| d 7} n q W| d j o
 | d j  Sd  S(   Ni    i   i   (	   s   ne0s   ne1s   succss   i0s   is   ss   removes   lens   e(   s   i0s   lens   ss   is   es   ne0s   ne1(   s   succs(    s*   /usr/lib/python2.2/test/test_generators.pys   remove_from_successorsùs   	 	c   s6   x) ˆ  |  D] } ˆ  | i |  ƒ q Wd  S(   N(   s   succss   i0s   is   append(   s   i0s   i(   s   succs(    s*   /usr/lib/python2.2/test/test_generators.pys   add_to_successorss    	c    #sq   ˆ d j  p
 ˆ d j  o d  Sn ˆ i d d ƒ }  ˆ  |  ƒ |  ˆ _ |  Vˆ |  ƒ d  S(   Ni   i    (   s   ms   ns   selfs   coords2indexs   corners   remove_from_successorss   lastijs   add_to_successors(   s   corner(   s   remove_from_successorss   selfs   ms   add_to_successorss   n(    s*   /usr/lib/python2.2/test/test_generators.pys   firsts   c    #s«  !"ˆ  i d d ƒ } #t o ˆ  i | j p t ‚ $ˆ d j  p
 ˆ d j  o %d  Sn &t o t ˆ | ƒ d j p t ‚ 't o$ ˆ  i d d ƒ ˆ | j p t ‚ (t o$ ˆ  i d d ƒ ˆ | j p t ‚ -x¼ d d f d d f f D-]Ÿ \ } } .ˆ  i | | ƒ }  /ˆ  i d | d | ƒ } 0| ˆ  _ 2ˆ |  ƒ 3ˆ | i | ƒ 4|  ˆ  _ 5|  V6ˆ | i | ƒ 7ˆ |  ƒ qWd  S(   Ni    i   i   i   (   s   selfs   coords2indexs   corners	   __debug__s   lastijs   AssertionErrors   ms   ns   lens   succss   is   js   thiss   finals   remove_from_successorss   appends   removes   add_to_successors(   s   thiss   js   is   corners   final(   s   selfs   ms   ns   succss   remove_from_successorss   add_to_successors(    s*   /usr/lib/python2.2/test/test_generators.pys   second!s$   !(.. c   #s  :=g  } >x™ ˆ  ˆ i D>]z } ?|  ˆ  | ƒ } @t o | d j p
 t d ‚ A| d j o B| | f g } CPn D| i	 | | f ƒ q WF| i
 ƒ  Hxb | DH]W \ } } I| ˆ i j o8 Jˆ | ƒ o K| ˆ _ L| Vn Mˆ | ƒ n qµ Wd  S(   Ni    s,   else remove_from_successors() pruning flawedi   (   s
   candidatess   succss   selfs   lastijs   is   lens   es	   __debug__s   AssertionErrors   appends   sorts   finals   remove_from_successorss   add_to_successors(   s   lens   is
   candidatess   e(   s   succss   remove_from_successorss   selfs   add_to_successors(    s*   /usr/lib/python2.2/test/test_generators.pys   advance:s"   	 	!
 i   f2.0c 	  #sV  TYg  } ZxÕ ˆ ˆ  i DZ]¶ } [| ˆ | ƒ } \t o | d j p
 t d ‚ ]| d j o ^| d | f g } _Pn `ˆ  i	 | ƒ \ } } a| |  d | | d } b| i | | | f ƒ q Wd| i ƒ  fxe | Df]Z \ } } } g| ˆ  i j o8 hˆ | ƒ o i| ˆ  _ j| Vn kˆ | ƒ n qñ Wd  S(   Ni    s,   else remove_from_successors() pruning flawedi   i   (   s
   candidatess   succss   selfs   lastijs   is   lens   es	   __debug__s   AssertionErrors   index2coordss   i1s   j1s   vmids   hmids   ds   appends   sorts   finals   remove_from_successorss   add_to_successors(	   s   vmids   hmids   lens   es   ds   i1s   j1s   is
   candidates(   s   selfs   remove_from_successorss   succss   add_to_successors(    s*   /usr/lib/python2.2/test/test_generators.pys   advance_hardTs&   	 	!
 c     #s9   not  o ˆ  i ˆ ˆ  i j p t ‚ pˆ  i Vd  S(   N(   s	   __debug__s   selfs   finals   succss   lastijs   AssertionError(    (   s   selfs   succs(    s*   /usr/lib/python2.2/test/test_generators.pys   lastns   (i   i   (   s   ms   ns   selfs   succss   lens   remove_from_successorss   add_to_successorss   firsts   seconds   advances   advance_hards   lasts   squaregeneratorss   hard(   s   selfs   ms   ns   hards   advances   lasts   seconds   succss   remove_from_successorss   add_to_successorss   advance_hards   first(    (   s   selfs   ms   ns   succss   remove_from_successorss   add_to_successorss*   /usr/lib/python2.2/test/test_generators.pys   __init__îs   1c   s}   yzt  o( d | j o |  i j  n p t ‚ {t  o( d | j o |  i j  n p t ‚ || |  i | Sd  S(   Ni    (   s	   __debug__s   is   selfs   ms   AssertionErrors   js   n(   s   selfs   is   j(    (    s*   /usr/lib/python2.2/test/test_generators.pys   coords2indexys   22c   sS   ~t  o/ d | j o |  i |  i j  n p t ‚ €t | |  i ƒ Sd  S(   Ni    (   s	   __debug__s   indexs   selfs   ms   ns   AssertionErrors   divmod(   s   selfs   index(    (    s*   /usr/lib/python2.2/test/test_generators.pys   index2coords~s   9c  	 st  ‚ƒ|  i }
 „|
 2…|  i |  i f \ } }	 †|  i } ˆd d f d d f d d f d d f d d f d d f d d f d d f g } Št |	 ƒ } ‹xÍ t | ƒ D‹]¼ } Œx° | DŒ]¥ } g  i } | D]r \ } } Žd | | j o
 | j  n o d | | j o
 |	 j  n o | | | | | | ƒ ƒ n qÚ ~ } |
 i | ƒ qÀ Wq­ Wd  S(   Ni   i   iÿÿÿÿiþÿÿÿi    (   s   selfs   succss   ms   ns   coords2indexs   c2is   offsetss   ranges   rangens   is   js   appends   _[1]s   ios   jos   s(   s   selfs   jos   offsetss   is   js   ms   ss   rangens   _[1]s   ns   succss   ios   c2i(    (    s*   /usr/lib/python2.2/test/test_generators.pys   _init_board‚s   Q 	
 	 oc   #s;   “”|  i ƒ  •x! t |  i ƒ D•] } –| Vq# Wd  S(   N(   s   selfs   _init_boards   conjoins   squaregeneratorss   x(   s   selfs   x(    (    s*   /usr/lib/python2.2/test/test_generators.pys   solve“s    	c   sz  ˜™|  i |  i f \ } } št o t | ƒ | | j p t ‚ ›t t | | ƒ ƒ } œd t | ƒ d } žg  i
 }	 t | ƒ Dž] } |	 t g | ƒ q ~	 } Ÿd }  xK | D ]@ } ¡|  i | ƒ \ }
 } ¢| | | |
 | <£| d 7} qÃ W¥d d | d | } ¦| GH§xE t | ƒ D§]4 } ¨| | } ©d d i | ƒ d GHª| GHq;Wd  S(   Ns   %s   di   s   +s   -s   |(   s   selfs   ms   ns	   __debug__s   lens   xs   AssertionErrors   strs   ws   formats   appends   _[1]s   ranges   is   Nones   squaress   ks   index2coordss   i1s   j1s   seps   rows   join(   s   selfs   xs   ws   formats   is   seps   ms   j1s   ns   _[1]s   i1s   squaress   ks   row(    (    s*   /usr/lib/python2.2/test/test_generators.pys   printsolution˜s&   ( #	
 	 	(   s   __name__s
   __module__s   __init__s   coords2indexs   index2coordss   _init_boards   solves   printsolution(    (    (    s*   /usr/lib/python2.2/test/test_generators.pys   Knightsís   	‹s$  

Generate the 3-bit binary numbers in order.  This illustrates dumbest-
possible use of conjoin, just to generate the full cross-product.

>>> for c in conjoin([lambda: iter((0, 1))] * 3):
...     print c
[0, 0, 0]
[0, 0, 1]
[0, 1, 0]
[0, 1, 1]
[1, 0, 0]
[1, 0, 1]
[1, 1, 0]
[1, 1, 1]

For efficiency in typical backtracking apps, conjoin() yields the same list
object each time.  So if you want to save away a full account of its
generated sequence, you need to copy its results.

>>> def gencopy(iterator):
...     for x in iterator:
...         yield x[:]

>>> for n in range(10):
...     all = list(gencopy(conjoin([lambda: iter((0, 1))] * n)))
...     print n, len(all), all[0] == [0] * n, all[-1] == [1] * n
0 1 1 1
1 2 1 1
2 4 1 1
3 8 1 1
4 16 1 1
5 32 1 1
6 64 1 1
7 128 1 1
8 256 1 1
9 512 1 1

And run an 8-queens solver.

>>> q = Queens(8)
>>> LIMIT = 2
>>> count = 0
>>> for row2col in q.solve():
...     count += 1
...     if count <= LIMIT:
...         print "Solution", count
...         q.printsolution(row2col)
Solution 1
+-+-+-+-+-+-+-+-+
|Q| | | | | | | |
+-+-+-+-+-+-+-+-+
| | | | |Q| | | |
+-+-+-+-+-+-+-+-+
| | | | | | | |Q|
+-+-+-+-+-+-+-+-+
| | | | | |Q| | |
+-+-+-+-+-+-+-+-+
| | |Q| | | | | |
+-+-+-+-+-+-+-+-+
| | | | | | |Q| |
+-+-+-+-+-+-+-+-+
| |Q| | | | | | |
+-+-+-+-+-+-+-+-+
| | | |Q| | | | |
+-+-+-+-+-+-+-+-+
Solution 2
+-+-+-+-+-+-+-+-+
|Q| | | | | | | |
+-+-+-+-+-+-+-+-+
| | | | | |Q| | |
+-+-+-+-+-+-+-+-+
| | | | | | | |Q|
+-+-+-+-+-+-+-+-+
| | |Q| | | | | |
+-+-+-+-+-+-+-+-+
| | | | | | |Q| |
+-+-+-+-+-+-+-+-+
| | | |Q| | | | |
+-+-+-+-+-+-+-+-+
| |Q| | | | | | |
+-+-+-+-+-+-+-+-+
| | | | |Q| | | |
+-+-+-+-+-+-+-+-+

>>> print count, "solutions in all."
92 solutions in all.

And run a Knight's Tour on a 10x10 board.  Note that there are about
20,000 solutions even on a 6x6 board, so don't dare run this to exhaustion.

>>> k = Knights(10, 10)
>>> LIMIT = 2
>>> count = 0
>>> for x in k.solve():
...     count += 1
...     if count <= LIMIT:
...         print "Solution", count
...         k.printsolution(x)
...     else:
...         break
Solution 1
+---+---+---+---+---+---+---+---+---+---+
|  1| 58| 27| 34|  3| 40| 29| 10|  5|  8|
+---+---+---+---+---+---+---+---+---+---+
| 26| 35|  2| 57| 28| 33|  4|  7| 30| 11|
+---+---+---+---+---+---+---+---+---+---+
| 59|100| 73| 36| 41| 56| 39| 32|  9|  6|
+---+---+---+---+---+---+---+---+---+---+
| 74| 25| 60| 55| 72| 37| 42| 49| 12| 31|
+---+---+---+---+---+---+---+---+---+---+
| 61| 86| 99| 76| 63| 52| 47| 38| 43| 50|
+---+---+---+---+---+---+---+---+---+---+
| 24| 75| 62| 85| 54| 71| 64| 51| 48| 13|
+---+---+---+---+---+---+---+---+---+---+
| 87| 98| 91| 80| 77| 84| 53| 46| 65| 44|
+---+---+---+---+---+---+---+---+---+---+
| 90| 23| 88| 95| 70| 79| 68| 83| 14| 17|
+---+---+---+---+---+---+---+---+---+---+
| 97| 92| 21| 78| 81| 94| 19| 16| 45| 66|
+---+---+---+---+---+---+---+---+---+---+
| 22| 89| 96| 93| 20| 69| 82| 67| 18| 15|
+---+---+---+---+---+---+---+---+---+---+
Solution 2
+---+---+---+---+---+---+---+---+---+---+
|  1| 58| 27| 34|  3| 40| 29| 10|  5|  8|
+---+---+---+---+---+---+---+---+---+---+
| 26| 35|  2| 57| 28| 33|  4|  7| 30| 11|
+---+---+---+---+---+---+---+---+---+---+
| 59|100| 73| 36| 41| 56| 39| 32|  9|  6|
+---+---+---+---+---+---+---+---+---+---+
| 74| 25| 60| 55| 72| 37| 42| 49| 12| 31|
+---+---+---+---+---+---+---+---+---+---+
| 61| 86| 99| 76| 63| 52| 47| 38| 43| 50|
+---+---+---+---+---+---+---+---+---+---+
| 24| 75| 62| 85| 54| 71| 64| 51| 48| 13|
+---+---+---+---+---+---+---+---+---+---+
| 87| 98| 89| 80| 77| 84| 53| 46| 65| 44|
+---+---+---+---+---+---+---+---+---+---+
| 90| 23| 92| 95| 70| 79| 68| 83| 14| 17|
+---+---+---+---+---+---+---+---+---+---+
| 97| 88| 21| 78| 81| 94| 19| 16| 45| 66|
+---+---+---+---+---+---+---+---+---+---+
| 22| 91| 96| 93| 20| 69| 82| 67| 18| 15|
+---+---+---+---+---+---+---+---+---+---+
s   tuts   peps   emails   funs   syntaxs   conjoinc   s;   JKd  k  } d  k } d  k } LR| i | |  ƒ d  S(   N(   s   doctests   test_supports   test_generatorss   run_doctests   verbose(   s   verboses   test_supports   test_generatorss   doctest(    (    s*   /usr/lib/python2.2/test/test_generators.pys	   test_mainJs   s   __main__i   N(   s
   __future__s
   generatorss   tutorial_testss	   pep_testss   email_testss	   fun_testss   syntax_testss   conjoins   flat_conjoins   Queenss   Knightss   conjoin_testss   __test__s   Nones	   test_mains   __name__(   s	   fun_testss   conjoins   syntax_testss   conjoin_testss   email_testss   tutorial_testss   flat_conjoins	   pep_testss	   test_mains   __test__s   Knightss   Queenss
   generators(    (    s*   /usr/lib/python2.2/test/test_generators.pys   ? s   	‡	¥	¼	 	µ=$7¿	“?