66

ダウンロードすると結果が出る関数があります。この質問の目的のために、1秒に1回刺し傷を出すが、ジェネレーターをラップする便利な関数が必要だとします。

import time

def GeneratorFunction(max_val):
    for i in range(0,5):
        time.sleep(1)
        yield "String %d"%i

def SmallGenerator():
    yield GeneratorFunction(3)

for s in SmallGenerator():
    print s

...なぜそれは私が期待している5つの文字列を印刷しないのですか?代わりに、ジェネレータ関数を返すように見えます。

<generator object GeneratorFunction at 0x020649B8>

通常のジェネレーター関数のように文字列を生成するためにこれを取得するにはどうすればよいですか?

4

5 に答える 5

56

「<ahref="https://docs.python.org/3/reference/expressions.html#yieldexpr"rel = "noreferrer">委任されたジェネレーター」として知られる、Python3.3以降で利用可能な新しいyield fromを使用する必要がある場合があります。 。

質問を正しく理解すれば、同じ問題に遭遇し、他の場所で答えを見つけました。

私はこのようなことをしたかった:

def f():

    def g():

        do_something()
        yield x
        …
        yield y

    do_some_other_thing()
    yield a
    …
    g()  # Was not working.
    yield g()  # Was not what was expected neither; yielded None.
    …
    yield b

代わりにこれを使用します:

yield from g()  # Now it works, it yields x and Y.

私はこのページから答えを得ました:Python 3:ジェネレーターでの「yieldfrom」の使用-パート1(simeonvisser.com)

于 2014-09-05T13:18:51.353 に答える
37

私がこれを逃したなんて信じられない。答えは、適切な引数が適用されたジェネレーター関数を返すことです。

import time

def GeneratorFunction(max_val):
    for i in range(0,max_val):
        time.sleep(1)
        yield "String %d"%i

def SmallGenerator():
    return GeneratorFunction(3) # <-- note the use of return instead of yield

for s in SmallGenerator():
    print s
于 2012-06-25T20:54:24.177 に答える
5

別の形の「入れ子になった収穫」を探しにここに来て、ついに隠された答えを見つけました。最高ではないかもしれませんが、それは機能します。

私はレジストリツリーを介して譲歩したかったのですが、これが解決策です。

        def genKeys(key):
            for value in key.values():
                yield value
            for subkey in key.subkeys():
                print(subkey)
                for x in genKeys(subkey): #this is the trick
                    continue
于 2016-07-22T16:09:03.583 に答える
1

1から10までの掛け算の九九を生成するための別の小さな例を次に示します。

class Gen1:

    def __init__(self, gen2):
        self.gen2 = gen2

    def __iter__(self):    
        for a in range(1, 11):    
            for b in self.gen2:
                yield a * b


class Gen2:    

    def __iter__(self):
        for a in range(1, 11):
            yield a


def main():

    gen2 = Gen2()
    gen1 = Gen1(gen2)

    for v in gen1:
        print(v)

if __name__ == '__main__':
    main()
于 2017-06-22T21:09:34.153 に答える
1

ネストされたyieldの別の使用法を探しに来ました。

list_of_lists = [
    [1, 2, 3],
    [4, 5, 6],
    [7, 8, 9],
]

def iter_all(foo):
    yield foo

    if isinstance(foo, list):
        for i in foo:
            for x in iter_all(i):
                yield x


print([i for i in iter_all(list_of_lists)])

出力:

[[[1, 2, 3], [4, 5, 6], [7, 8, 9]], [1, 2, 3], 1, 2, 3, [4, 5, 6], 4, 5, 6, [7, 8, 9], 7, 8, 9]
于 2019-02-12T13:43:15.647 に答える