turtle.delay()

2025-06-06

turtle.delay()とは何か?

通常、turtleで線を描いたり図形を動かしたりする際、デフォルトでは非常に速く描画されます。turtle.delay(n)と記述すると、各タートルアクション(例えば、一歩進む、角度を変えるなど)の間にnミリ秒(1000分の1秒)の遅延が挿入されます。

なぜ使うのか?

  • パフォーマンスへの影響
    非常に小さい遅延(例えば、delay(1)delay(0))を設定すると、描画処理がよりスムーズに見える場合があります。ただし、delay(0)は実際には可能な限り速く描画するという意味になり、ほとんど遅延がない状態になります。
  • 視覚的な効果
    アニメーションに意図的に「間」を持たせることで、より自然な動きや特定の視覚的効果を生み出すことができます。
  • アニメーションの観察
    アニメーションが速すぎて何が起こっているのか分かりにくい場合に、遅延を入れることで動きをよりゆっくりと見ることができます。デバッグや学習に役立ちます。

使用例

import turtle

t = turtle.Turtle()
screen = turtle.Screen()

# デフォルトでは非常に速く描画されます
# t.forward(100)
# t.left(90)
# t.forward(100)

# 200ミリ秒(0.2秒)の遅延を設定
screen.delay(200) # または turtle.delay(200)

t.forward(100)
t.left(90)
t.forward(100)
t.left(90)
t.forward(100)
t.left(90)
t.forward(100)

turtle.done()

上記の例では、四角形を描画する各ステップの間で200ミリ秒の遅延が発生するため、四角形がゆっくりと描画されるのが分かります。

  • delay()は通常、turtle.Screen()オブジェクトのメソッドとして呼び出すか、直接turtleモジュールから呼び出すことができます(screen.delay(n)またはturtle.delay(n))。どちらも同じ効果があります。
  • delay(0)を設定すると、描画は可能な限り速く行われます。これは、事実上遅延がない状態を意味します。
  • delay()に設定する値はミリ秒単位です。


turtle.delay()はタートルの描画動作の間にミリ秒単位の遅延を設定するものです。しかし、以下のような状況で問題が発生することがあります。

AttributeError: 'Turtle' object has no attribute 'delay' または AttributeError: 'RawTurtle' object has no attribute 'delay'

これは最もよくあるエラーの一つです。delay()メソッドは、turtle.Screen()オブジェクトまたはturtleモジュール自体に属しています。特定のTurtleオブジェクト(例えばt = turtle.Turtle()で作成したt)には直接delay()メソッドは存在しません。

間違った例

import turtle
t = turtle.Turtle()
t.delay(100) # エラーになる

解決策
turtle.Screen()オブジェクトを使って遅延を設定します。

import turtle
screen = turtle.Screen() # スクリーンオブジェクトを作成
t = turtle.Turtle()

screen.delay(100) # 正しい使い方:スクリーンオブジェクトのdelayメソッドを呼び出す

t.forward(50)
t.left(90)
t.forward(50)

turtle.done()

または、turtleモジュールから直接呼び出すこともできます(内部的にはスクリーンの設定に影響します)。

import turtle

t = turtle.Turtle()

turtle.delay(100) # 正しい使い方:turtleモジュールから直接呼び出す

t.forward(50)
t.left(90)
t.forward(50)

turtle.done()

delay()を設定してもアニメーションが速すぎる、または遅すぎる

delay()の値の解釈に誤解がある場合があります。

  • 他の速度設定との競合
    turtle.speed()もアニメーション速度に影響します。speed(0)は「最速」を意味し、通常delay()による設定を上書きする可能性があります。speed(0)を設定している場合、delay()の効果が見られないことがあります。

    トラブルシューティング

    t.speed(0)turtle.speed(0)を使っている場合は、これを削除するか、t.speed(1)などの具体的な速度値に設定してみてください。delay()speed()は、どちらもアニメーションの速度を制御する機能であり、相互に影響し合います。両方を同時に設定している場合、意図しない挙動になることがあります。

  • 値の単位がミリ秒であることの確認
    delay(1000)は1秒の遅延ですが、delay(1)は1ミリ秒の遅延であり、ほとんど感知できないほど速いです。

delay()を設定しても全く効果がない

  • プログラムが即座に終了してしまう
    スクリプトの最後にturtle.done()screen.mainloop()がない場合、タートルのウィンドウがすぐに閉じてしまい、アニメーションを見る前にプログラムが終了してしまうことがあります。

    トラブルシューティング

    スクリプトの最後にturtle.done()またはscreen.mainloop()を追加して、ウィンドウが開いたままになるようにしてください。

  • turtle.tracer()の使用
    turtle.tracer(0)を設定している場合、描画の更新がオフになり、アニメーションが非常に高速になります。この場合、delay()は効果を発揮しません。

    トラブルシューティング

    turtle.tracer(0)を使用している場合は、描画を有効にするためにscreen.update()を適宜呼び出すか、turtle.tracer(1)に戻してみてください。

    import turtle
    
    screen = turtle.Screen()
    t = turtle.Turtle()
    
    screen.tracer(0) # 描画をオフにする
    screen.delay(500) # 遅延を設定しても、tracer(0)が優先されるため効果がない
    
    for _ in range(4):
        t.forward(100)
        t.left(90)
        screen.update() # tracer(0)を使用する場合は手動で更新が必要
    
    screen.tracer(1) # tracerを有効に戻す
    screen.delay(500) # ここからdelayが効き始める
    
    t.forward(50)
    
    turtle.done()
    

実行環境による挙動の違い

  • IDEやオンライン環境
    一部のIDE(例: PyCharm)やオンラインのPython実行環境(Repl.it, Trinketなど)では、turtleの描画挙動がローカルPCでの実行と異なる場合があります。これは、それらの環境が描画処理を最適化したり、特定の制約を設けたりするためです。

    トラブルシューティング

    もしオンライン環境で問題が発生している場合は、ローカルのPython環境で試してみるか、その環境のドキュメントを確認してみてください。

    また、キー入力の繰り返し遅延など、turtle自体ではなくOSのシステム設定に起因する遅延の場合もあります。

大量の回転(left() / right())による不必要な遅延

turtleの古いバージョンや特定の状況では、right(360 * 1000)のように非常に大きな角度でタートルを回転させると、実際の描画には影響がなくても、内部処理で不必要な遅延が発生することが報告されています。これは、アニメーションがオンになっている場合に、大きな角度が小さなステップに分割されて処理されるためです。

トラブルシューティング

  • 大きな角度で回転させる必要がある場合でも、−180°から180°の範囲に角度を正規化することを検討してください(例: angle % 360math.fmod(angle, 360)など)。
  • turtle.done()またはscreen.mainloop()の確認
    アニメーションの表示を確実にするために、これらの関数がプログラムの最後に呼び出されているか確認します。
  • print()デバッグ
    コードの各ステップで時間がどれだけかかっているかをtime.time()などを使って測定し、問題のボトルネックを特定します。
  • ドキュメントの参照
    turtleモジュールの公式ドキュメント(Pythonの公式サイト)を参照するのが最も確実です。
  • コードの簡素化
    問題が発生している部分だけを切り出して、簡単なコードで再現を試みます。
  • エラーメッセージをよく読む
    AttributeErrorが出た場合、どのオブジェクトがdelay()を持っていないのかを確認します。


turtle.delay()関数は、タートルグラフィックスのアニメーションの速度を制御するために使用されます。描画される各ステップの間に指定したミリ秒の遅延を挿入することで、アニメーションの様子を観察しやすくしたり、特定の視覚効果を作り出したりできます。

delay()の基本的な使い方 (正方形を描く)

最も基本的な例として、正方形をゆっくりと描画するコードを見てみましょう。

import turtle
import time # 必要に応じて時間の計測に使うこともできます

# 1. スクリーンオブジェクトの取得
screen = turtle.Screen()

# 2. タートルオブジェクトの作成
t = turtle.Turtle()

# 3. delay()を設定 (200ミリ秒 = 0.2秒の遅延)
# screen.delay(200) は turtle.delay(200) と同じ効果があります。
screen.delay(200) 

# 4. 正方形を描画
print("正方形を描画開始...")
for _ in range(4):
    t.forward(100) # 100ピクセル前進
    t.left(90)     # 左に90度回転
print("正方形を描画完了!")

# 5. ウィンドウを閉じるまで待機
turtle.done() 
# または screen.mainloop()

解説

  • turtle.done(): これがないと、Pythonスクリプトが終了すると同時にタートルのウィンドウも閉じてしまいます。この関数を呼び出すことで、ユーザーがウィンドウを閉じるまで表示され続けます。
  • screen.delay(200): ここがポイントです。タートルがforward(100)を実行し終えてから次のleft(90)を実行するまでの間、そしてleft(90)を実行し終えてから次のforward(100)を実行するまでの間、それぞれ200ミリ秒(0.2秒)の遅延が入ります。これにより、正方形が1辺ずつゆっくりと描画される様子が視覚的に確認できます。
  • t = turtle.Turtle(): 描画を行う「タートル」オブジェクトを作成します。
  • screen = turtle.Screen(): タートルの描画が行われる「スクリーン」オブジェクトを取得します。delay()メソッドはこのスクリーンオブジェクトに属します。

delay()の値を変更して速度を比較する

遅延時間を変えることで、アニメーションの速度がどのように変化するかを見てみましょう。

import turtle

screen = turtle.Screen()
screen.setup(width=600, height=400) # ウィンドウサイズを設定
screen.tracer(False) # 初期描画をオフにして準備を高速化

# 最初のタートル(遅い)
t1 = turtle.Turtle()
t1.penup()
t1.goto(-200, 0)
t1.pendown()
t1.speed(0) # 最速で描画を準備

# 2番目のタートル(速い)
t2 = turtle.Turtle()
t2.penup()
t2.goto(100, 0)
t2.pendown()
t2.speed(0) # 最速で描画を準備

screen.tracer(True) # 描画をオンに戻す

print("異なる遅延時間で描画を開始します。")

# t1は遅延が大きい
screen.delay(300) # 300ミリ秒 = 0.3秒の遅延
print("左側のタートルが遅延 300ms で描画中...")
for _ in range(4):
    t1.forward(100)
    t1.left(90)

# t2は遅延が小さい
screen.delay(50) # 50ミリ秒 = 0.05秒の遅延
print("右側のタートルが遅延 50ms で描画中...")
for _ in range(4):
    t2.forward(100)
    t2.left(90)

print("描画完了!")
turtle.done()

解説

  • この例では、同じscreen.delay()を呼び出すことで、両方のタートルに異なるタイミングで遅延設定が適用されます。screen.delay(300)を設定すると左のタートルがゆっくり動き、次にscreen.delay(50)を設定すると右のタートルが速く動くのが分かります。turtle.delay()はグローバルな設定であるため、一度設定するとすべてのタートルに影響します。
  • t1.speed(0): タートルのspeed()メソッドを0に設定すると、タートルは「最速」で動きます。ただし、delay()が設定されている場合は、delay()が優先されます。
  • screen.tracer(False): この行は重要です。描画を一時的に無効にすることで、タートルの初期位置設定などが瞬時に行われます。screen.tracer(True)を呼び出すまで、実際の描画は行われません。これは、多くの初期設定を行う場合にアニメーションをスキップして高速化するテクニックです。
  • screen.setup(width=600, height=400): ウィンドウのサイズを設定します。

tracer()とdelay()の組み合わせ

tracer(0)を使うとdelay()の効果は一時的に無効になりますが、screen.update()と組み合わせることで独自のフレーム制御が可能です。

import turtle

screen = turtle.Screen()
t = turtle.Turtle()
t.speed(0) # タートルの速度は最速に設定

# tracer(0)を設定すると、自動描画更新が無効になる
# この状態ではdelay()も効果を発揮しません。
screen.tracer(0) 

print("tracer(0)と手動更新で描画中...")

for i in range(100):
    t.circle(i) # 円を描く
    t.left(5)   # 少し回転
    
    # 20ミリ秒ごとに画面を更新
    # これにより、delay()とは異なる方法でアニメーション速度を制御します。
    screen.update() 
    time.sleep(0.02) # timeモジュールを使って20ミリ秒待機 (これは必須ではないが、より正確な遅延制御に役立つ)

print("描画完了!")
turtle.done()
  • この例ではturtle.delay()は直接使用していませんが、tracer()time.sleep()delay()の代わりとなるアニメーション制御の方法として示されています。
  • time.sleep(0.02): turtle.delay()がタートルの各ステップ間に遅延を挿入するのに対し、time.sleep()はPythonプログラム全体の実行を一時停止させます。この組み合わせにより、tracer(0)で高速に描画処理を行い、update()で一括表示し、その表示間の間隔をtime.sleep()で調整するといった、より細やかな制御が可能になります。
  • screen.update(): tracer(0)を設定している場合、この関数を呼び出すことで、バッファに蓄積された描画内容が一度に画面に表示されます。
  • screen.tracer(0): これにより、タートルが行うすべての描画操作が画面に即座に反映されなくなります。描画は内部のバッファに蓄積されます。


turtle.delay()はアニメーションの速度を制御する便利な関数ですが、より細かい制御や異なるタイプのアニメーションを実現したい場合には、いくつかの代替手段があります。

turtle.speed()の利用

turtle.speed()メソッドは、タートルの描画速度を制御する最も直接的な方法の一つです。delay()が「各ステップ間の遅延」を設定するのに対し、speed()は「描画アニメーションの速度」をより抽象的なレベルで制御します。

特徴

  • speed(0)は特別な意味を持ち、可能な限り速く描画されます。delay()が設定されていても、speed(0)が優先される場合が多いです。
  • 引数として0(最速)、1(最も遅い)から10(非常に速い)までの整数、または文字列("fastest", "fast", "normal", "slow", "slowest")を取ります。

使用例

import turtle

screen = turtle.Screen()
t = turtle.Turtle()

# 速度を'normal' (通常) に設定
t.speed('normal') 
print("速度'normal'で描画中...")
for _ in range(4):
    t.forward(100)
    t.left(90)

t.penup()
t.goto(0, -100)
t.pendown()

# 速度を'fastest' (最速) に設定
t.speed('fastest') 
print("速度'fastest'で描画中...")
for _ in range(4):
    t.forward(100)
    t.left(90)

turtle.done()

delay()との比較
speed()delay()よりも高レベルな制御です。delay()が個々の描画ステップの間にミリ秒単位のポーズを入れるのに対し、speed()はタートルの動き全体のアニメーション速度を調整します。通常、まずspeed()で大まかな速度を設定し、さらに微調整が必要な場合にdelay()を検討すると良いでしょう。ただし、speed(0)delay()の効果を打ち消すことが多いです。

turtle.tracer() と screen.update() の組み合わせ

これはturtleグラフィックスで最も強力で柔軟なアニメーション制御方法の一つです。turtle.tracer()は、タートルの描画が自動的に画面に反映されるかどうかを制御します。

特徴

  • screen.update(): tracer(0)が設定されている場合に、バッファに蓄積されたすべての描画を一度に画面に表示します。
  • screen.tracer(n): nフレームごとに画面を更新します。n=0を設定すると、自動的な画面更新が完全に停止します。

メリット

  • time.sleep()と組み合わせることで、フレームレートを自由に制御できます。
  • 複雑なアニメーション(多数のタートルや多数のステップ)をスムーズに表示するのに適しています。
  • 描画処理を非常に高速に行い、描画結果だけを希望のタイミングで表示できます。

使用例

import turtle
import time

screen = turtle.Screen()
t = turtle.Turtle()
t.speed(0) # tracerを使用するので、タートルのspeedは最速でOK
screen.setup(width=600, height=600)

# 自動画面更新をオフにする (非常に重要!)
screen.tracer(0) 

print("tracer(0)と手動更新で複雑な描画を開始...")

for i in range(360): # 360ステップのアニメーション
    t.forward(2)
    t.left(1)
    if i % 10 == 0: # 10ステップごとに画面を更新
        screen.update()
        time.sleep(0.01) # 10ミリ秒待機してフレームレートを調整

# 最後にすべての変更を更新
screen.update()
print("描画完了!")
turtle.done()

time.sleep()の直接利用

Pythonの標準ライブラリであるtimeモジュールのtime.sleep()関数は、プログラムの実行を指定した秒数だけ一時停止させます。

特徴

  • プログラム全体の実行をブロックします。
  • 引数は秒単位です(浮動小数点数も可)。

メリット

  • タートルの描画だけでなく、他のPythonコードの実行も一時停止させたい場合に便利。
  • シンプルで分かりやすい。

使用例

import turtle
import time

screen = turtle.Screen()
t = turtle.Turtle()
t.speed(1) # 遅めのspeed設定にして、time.sleepの効果を強調

print("time.sleep()でアニメーションを制御...")

for _ in range(4):
    t.forward(100)
    print("前進しました。")
    time.sleep(0.5) # 0.5秒(500ミリ秒)プログラム全体を一時停止
    t.left(90)
    print("回転しました。")
    time.sleep(0.5) # 0.5秒プログラム全体を一時停止

print("描画完了!")
turtle.done()

delay()との比較
delay()はタートルの描画イベントの間に自動的に遅延を挿入しますが、time.sleep()はコードの好きな場所に明示的に配置して、その場所でプログラムの実行を一時停止させます。よりきめ細かい制御が可能ですが、delay()のように自動的に各ステップに適用されるわけではないため、自分でsleep()を挿入する必要があります。通常、tracer(0)と組み合わせて、update()の後にtime.sleep()を使うのが一般的です。

  • time.sleep(): プログラムの特定の時点で実行を一時停止させたい場合に便利。tracer()と組み合わせてフレームレート制御に使うことが多い。
  • turtle.tracer() + screen.update(): 複雑で高性能なアニメーション、細かいフレームレート制御が必要な場合に最適。最も柔軟。
  • turtle.speed(): アニメーション速度の大まかな調整に最適。最も簡単。