歩留まりについての深い理解

大家讲道理
リリース: 2016-11-07 16:45:36
オリジナル
1186 人が閲覧しました

英語の yield は生産を意味します。初めて Python に触れたとき、私は非常に混乱し、yield の使い方がわかりませんでした。

次の例のように、 yield を使用して関数の戻り値にデータを詰め込むことができることは大まかに知っています:

def addlist(alist):
    for i in alist:
        yield i + 1
ログイン後にコピー



alist の各項目を取り出し、 i + 1 を詰めます。そこに。次に、呼び出して各項目を取り出します:

alist = [1, 2, 3, 4]
for x in addlist(alist):
    print x,
ログイン後にコピー

これは実際に yield アプリケーションの例です

1. yield を含む関数

を含む関数が表示された場合、この関数はすでにジェネレーターであることを意味します。他の通常の関数とは大きく異なります。たとえば、次の単純な関数:

def h():
    print 'To be brave'
    yield 5
h()
ログイン後にコピー

h() を呼び出した後、print ステートメントが実行されていないことがわかります。これは yield ですが、print ステートメントを実行するにはどうすればよいでしょうか?これについては後で説明します。この後の議論と検討を通じて、収量がどのように機能するかが理解できるようになります。

2. Yield は式です

Python 2.5 より前では、yield はステートメントでしたが、Python 2.5 では、yield は次のような式 (Expression) になりました。

m = yield 5

expression (yield 5 )は m に代入されるため、m = 5 と考えるのは間違いです。では、戻り値 (yield 5) を取得するにはどうすればよいでしょうか?後で紹介する send(msg) メソッドを使用する必要があります。

3. next() ステートメントを通して原理を見てみましょう

それでは、収量の動作原理を明らかにしましょう。上記の h() は、yield 式を持っているため、呼び出された後に実行されなかったことがわかっているため、next() ステートメントを通じて実行させます。 next() ステートメントは、次の yield 式までジェネレーターの実行を再開します。例:

def h():
    print 'Wen Chuan'
    yield 5
    print 'Fighting!'
c = h()
c.next()
ログイン後にコピー

c.next() が呼び出された後、h() は yield 5 に到達するまで実行を開始します。そのため、出力結果は次のようになります:

Wen Chuan

c.next() を再度呼び出すと、実行は次のようになります。次の yield 式が見つかるまで続行します。後で利回りがないため、例外がスローされます:

Wen Chuan

Fighting!

Traceback (most last call last):

File "/home/evergreen/Codes/yidld.py", line 11, in

c.next()

StopIteration

4. send(msg) と next()

next() によって、yield を含む関数がどのように実行されるかを理解した後、別の非常に重要な関数 send(メッセージ)。実際、 next() と send() は、ある意味では同様の関数を持っています。違いは、send() は yield 式の値を渡すことができるのに対し、next() は特定の値を渡すことができず、None のみを渡すことができるということです。したがって、

c.next() と c.send(None) は同じ効果があることがわかります。

この例を見てください:

def h():
    print 'Wen Chuan',
    m = yield 5  # Fighting!
    print m
    d = yield 12
    print 'We are together!'
c = h()
c.next()  #相当于c.send(None)
c.send('Fighting!')  #(yield 5)表达式被赋予了'Fighting!'
ログイン後にコピー

出力結果は次のとおりです:

Wen Chuan Fighting!

初めて呼び出すときは、next() ステートメントまたは send(None) を使用してください。 use send None 以外の値を送信します。送信しないと、この値を受け取る yield ステートメントがないため、エラーが発生します。

5. send(msg) と next() の戻り値

send(msg) と next() の戻り値は、次の yield 式のパラメーターを返します。たとえば、yield 5 の場合、5 が返されます。ここで何か理解できましたか?この記事の最初の例では、alist の for i を介してジェネレーターをトラバースすることにより、alist.Next() が実際に毎回呼び出され、毎回の alist.Next() の戻り値が yield のパラメーターになります。何かが押し込まれているのではないかと思い始めます。上記の例を続けてみましょう:

def h():
    print 'Wen Chuan',
    m = yield 5  # Fighting!
    print m
    d = yield 12
    print 'We are together!'
c = h()
m = c.next()  #m 获取了yield 5 的参数值 5
d = c.send('Fighting!')  #d 获取了yield 12 的参数值12
print 'We will never forget the date', m, '.', d
ログイン後にコピー

出力結果:

Wen Chuan Fighting!

日付は決して忘れません 5. 12

6. throw() と close() がジェネレーターを中断します

ジェネレーターを中断するのは次のとおりです。 GeneratorExit 例外をスローしてジェネレーターを終了する非常に柔軟な手法です。実際、Close() メソッドには同じ機能があり、内部で throw(GeneratorExit) を呼び出します。見てみましょう:

def close(self):
    try:
        self.throw(GeneratorExit)
    except (GeneratorExit, StopIteration):
        pass
    else:
        raise RuntimeError("generator ignored GeneratorExit")
# Other exceptions are not caught
ログイン後にコピー

したがって、close() メソッドを呼び出してから next() または send(msg) を呼び出すと、例外がスローされます:

Traceback (most recent call last):
  File "/home/evergreen/Codes/yidld.py", line 14, in <module>
    d = c.send(&#39;Fighting!&#39;)  #d 获取了yield 12 的参数值12
StopIteration
ログイン後にコピー


関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!