995

Python プログラムで do-while ループをエミュレートする必要があります。残念ながら、次の簡単なコードは機能しません。

list_of_ints = [ 1, 2, 3 ]
iterator = list_of_ints.__iter__()
element = None

while True:
  if element:
    print element

  try:
    element = iterator.next()
  except StopIteration:
    break

print "done"

「1,2,3,done」の代わりに、次の出力を出力します。

[stdout:]1
[stdout:]2
[stdout:]3
None['Traceback (most recent call last):
', '  File "test_python.py", line 8, in <module>
    s = i.next()
', 'StopIteration
']

「反復停止」例外をキャッチして while ループを適切に中断するにはどうすればよいですか?

そのようなことが必要な理由の例を、疑似コードとして以下に示します。

ステート マシン:

s = ""
while True :
  if state is STATE_CODE :
    if "//" in s :
      tokens.add( TOKEN_COMMENT, s.split( "//" )[1] )
      state = STATE_COMMENT
    else :
      tokens.add( TOKEN_CODE, s )
  if state is STATE_COMMENT :
    if "//" in s :
      tokens.append( TOKEN_COMMENT, s.split( "//" )[1] )
    else
      state = STATE_CODE
      # Re-evaluate same line
      continue
  try :
    s = i.next()
  except StopIteration :
    break
4

19 に答える 19

1226

あなたが何をしようとしているのかわからない。次のような do-while ループを実装できます。

while True:
  stuff()
  if fail_condition:
    break

または:

stuff()
while not fail_condition:
  stuff()

do while ループを使用してリスト内のものを印刷しようとしているのは何ですか? なぜ使用しないのですか:

for i in l:
  print i
print "done"

アップデート:

それで、あなたは行のリストを持っていますか?そして、あなたはそれを繰り返し続けたいですか?どうですか:

for s in l: 
  while True: 
    stuff() 
    # use a "break" instead of s = i.next()

それはあなたが望むものに近いもののように思えますか?コード例では、次のようになります。

for s in some_list:
  while True:
    if state is STATE_CODE:
      if "//" in s:
        tokens.add( TOKEN_COMMENT, s.split( "//" )[1] )
        state = STATE_COMMENT
      else :
        tokens.add( TOKEN_CODE, s )
    if state is STATE_COMMENT:
      if "//" in s:
        tokens.append( TOKEN_COMMENT, s.split( "//" )[1] )
        break # get next s
      else:
        state = STATE_CODE
        # re-evaluate same line
        # continues automatically
于 2009-04-13T06:28:43.760 に答える
365

do-while ループをエミュレートする非常に簡単な方法を次に示します。

condition = True
while condition:
    # loop body here
    condition = test_loop_condition()
# end of loop

do-while ループの主な特徴は、ループ本体が常に少なくとも 1 回実行されることと、ループ本体の最後で条件が評価されることです。ここに示す制御構造は、例外や break ステートメントを必要とせずに、これらの両方を実現します。追加のブール変数が 1 つ導入されます。

于 2010-03-14T00:09:54.823 に答える
91

以下の私のコードは、私が理解しているように、 whileの主な違いを強調しており、有用な実装になる可能性があります。

したがって、この 1 つのケースでは、必ずループを少なくとも 1 回通過します。

first_pass = True
while first_pass or condition:
    first_pass = False
    do_stuff()
于 2014-11-23T23:37:17.577 に答える
37
do {
  stuff()
} while (condition())

->

while True:
  stuff()
  if not condition():
    break

次の機能を実行できます。

def do_while(stuff, condition):
  while condition(stuff()):
    pass

しかし 1) 醜い。2) 条件は、1 つのパラメーターを持つ関数で、何かで埋められるはずです (これが、従来の while ループを使用しない唯一の理由です)。

于 2009-04-13T13:57:02.997 に答える
17

コルーチンを使用した、別のパターンのよりクレイジーなソリューションを次に示します。コードは非常によく似ていますが、重要な違いが 1 つあります。終了条件は一切ありません!コルーチン (実際にはコルーチンのチェーン) は、データのフィードを停止すると停止します。

def coroutine(func):
    """Coroutine decorator

    Coroutines must be started, advanced to their first "yield" point,
    and this decorator does this automatically.
    """
    def startcr(*ar, **kw):
        cr = func(*ar, **kw)
        cr.next()
        return cr
    return startcr

@coroutine
def collector(storage):
    """Act as "sink" and collect all sent in @storage"""
    while True:
        storage.append((yield))

@coroutine      
def state_machine(sink):
    """ .send() new parts to be tokenized by the state machine,
    tokens are passed on to @sink
    """ 
    s = ""
    state = STATE_CODE
    while True: 
        if state is STATE_CODE :
            if "//" in s :
                sink.send((TOKEN_COMMENT, s.split( "//" )[1] ))
                state = STATE_COMMENT
            else :
                sink.send(( TOKEN_CODE, s ))
        if state is STATE_COMMENT :
            if "//" in s :
                sink.send(( TOKEN_COMMENT, s.split( "//" )[1] ))
            else
                state = STATE_CODE
                # re-evaluate same line
                continue
        s = (yield)

tokens = []
sm = state_machine(collector(tokens))
for piece in i:
    sm.send(piece)

上記のコードは、すべてのトークンをタプルとして収集します。元のコードとtokensの間に違いはないと思います。.append().add()

于 2009-11-02T17:32:02.063 に答える
17

私がこれを行った方法は次のとおりです...

condition = True
while condition:
     do_stuff()
     condition = (<something that evaluates to True or False>)

これは単純な解決策のように思えますが、ここでまだ見ていないことに驚いています。これは明らかに逆にすることもできます

while not condition:

于 2018-06-23T20:00:26.973 に答える
10

for a do - while ループには try ステートメントが含まれています

loop = True
while loop:
    generic_stuff()
    try:
        questionable_stuff()
#       to break from successful completion
#       loop = False  
    except:
        optional_stuff()
#       to break from unsuccessful completion - 
#       the case referenced in the OP's question
        loop = False
   finally:
        more_generic_stuff()

あるいは、'finally' 句が必要ない場合

while True:
    generic_stuff()
    try:
        questionable_stuff()
#       to break from successful completion
#       break  
    except:
        optional_stuff()
#       to break from unsuccessful completion - 
#       the case referenced in the OP's question
        break
于 2011-05-10T22:03:06.030 に答える
8

この Python での do-while シミュレーションは、C や Java に存在する do-while 構造体の形式に最も近い構文形式を持っていると思います。

do = True
while do:
    [...]
    do = <condition>
于 2021-05-02T19:17:59.627 に答える
7

クイックハック:

def dowhile(func = None, condition = None):
    if not func or not condition:
        return
    else:
        func()
        while condition():
            func()

次のように使用します。

>>> x = 10
>>> def f():
...     global x
...     x = x - 1
>>> def c():
        global x
        return x > 0
>>> dowhile(f, c)
>>> print x
0
于 2013-04-21T21:42:57.920 に答える
7
while condition is True: 
  stuff()
else:
  stuff()
于 2010-11-30T01:38:03.477 に答える
4

やってみませんか

for s in l :
    print s
print "done"

?

于 2009-04-13T06:23:44.537 に答える
1

私の場合、典型的な while ループは次のようになります。

xBool = True
# A counter to force a condition (eg. yCount = some integer value)

while xBool:
    # set up the condition (eg. if yCount > 0):
        (Do something)
        yCount = yCount - 1
    else:
        # (condition is not met, set xBool False)
        xBool = False

状況に応じて、while ループ内にfor..loopを含めることもできます。これは、別の一連の条件をループするために必要です。

于 2020-03-08T10:08:22.363 に答える
1

これが役立つかどうかを確認してください:

例外ハンドラ内にフラグを設定し、s で作業する前に確認します。

flagBreak = false;
while True :

    if flagBreak : break

    if s :
        print s
    try :
        s = i.next()
    except StopIteration :
        flagBreak = true

print "done"
于 2009-04-13T08:17:55.697 に答える