锯齿 tkinter 主循环帧持续时间?

Posted

技术标签:

【中文标题】锯齿 tkinter 主循环帧持续时间?【英文标题】:Sawtooth tkinter mainloop frame duration? 【发布时间】:2012-10-06 02:29:06 【问题描述】:

尝试使用 tkinter 为 PIL 图像序列制作动画。我的帧持续时间(毫秒)图如下所示:

有人知道是什么导致了这种尖尖的锯齿图案吗?

这是一个重现的脚本:

from PIL import Image, ImageTk
import Tkinter

import time
import sys

def generate_frames(n):
    """
    keep n under 101 * 101
    """
    out = []
    last_pil = None
    for i in range(n):
        if last_pil:
            pil_image = last_pil.copy()
        else:
            pil_image = Image.new('L', (101, 101), 255)   
        x = i / 101
        y = i % 101
        pil_image.load()[x, y] = 0
        out.append(ImageTk.PhotoImage(pil_image))
        last_pil = pil_image

    return out

def draw():
    FRAME_COUNT =5000

    master = Tkinter.Tk()

    w = Tkinter.Canvas(master, width=302, height=302)
    w.create_rectangle(49, 49, 252, 252)
    w.pack()

    frames = generate_frames(FRAME_COUNT)

    def draw_frame(f, canvas_image):
        print repr(time.time())
        frame = frames[f]
        if canvas_image is None:
            canvas_image = w.create_image((151, 151), image=frame, anchor='center')
        else:
            w.itemconfigure(canvas_image, image=frame)

        w.current_frame = frame  # save a reference
        next_frame = f + 1
        if next_frame < FRAME_COUNT:
            master.after(1, draw_frame, next_frame, canvas_image)
        else:
            sys.exit(0)

    master.after(10, draw_frame, 0, None)
    master.mainloop()


draw()

要查看绘图,请通过管道输出

import sys

last = None
for line in sys.stdin:
    value = float(line.strip()) * 1000
    if last is None:
        pass
    else:
        print (value - last)
    last = value

然后通过

from matplotlib import pyplot
import sys

X = []
Y = []

for index, line in enumerate(sys.stdin):
    line = line.strip()
    X.append(index)
    Y.append(float(line))

pyplot.plot(X, Y, '-')
pyplot.show()

使其成为多线程并没有帮助:

class AnimationThread(threading.Thread):

    FRAME_COUNT = 5000

    def __init__(self, canvas):
        threading.Thread.__init__(self)
        self.canvas = canvas
        self.frames = generate_frames(self.FRAME_COUNT)

    def run(self):
        w = self.canvas
        frames = self.frames
        canvas_image = None
        for i in range(self.FRAME_COUNT):
            print repr(time.time())
            frame = frames[i]
            if canvas_image is None:
                canvas_image = w.create_image((151, 151), image=frame, anchor='center')
            else:
                w.itemconfigure(canvas_image, image=frame)
            w.current_frame = frame
            time.sleep(1 * .001)

def draw_threaded():
    FRAME_COUNT = 5000
    master = Tkinter.Tk()

    w = Tkinter.Canvas(master, width=302, height=302)
    w.create_rectangle(49, 49, 252, 252)
    w.pack()

    animation_thread = AnimationThread(w)
    animation_thread.start()

    master.mainloop()

    animation_thread.join()

draw_threaded()

【问题讨论】:

我会尝试在单独的线程中运行动画,而不是在 tkinter 主循环中,看看会发生什么。 好主意 - 谢谢。但这并没有什么不同(请参阅编辑后的帖子) 你试过用python -m cprofile &lt;script-name&gt;分析它吗? 您是否考虑过更改动画循环执行其帧的方式?而不是do-stuff-that-takes-variable-time =&gt; wait-for-fixed-time,您应该尝试do-stuff-async =&gt; if-time-is-right-continue-else-skip 这很可能是由于垃圾回收周期 - 你通常会得到这种模式。不过,我不确定你能做些什么来帮助它。 【参考方案1】:

当相互竞争的 60 赫兹和 50 赫兹样本混合时,这与这种干扰模式非常相似:

(Original Wolfram|Alpha plot)

这可能是由于两个事物的刷新率不同(但接近)造成的。当您尝试拍摄电视屏幕时会发生相同类型的事情,它看起来像一个黑条一直在图像中向下移动,或者当汽车广告中的车轮似乎围绕它们的轴向后旋转时。它本质上是Moiré Effect 的扩展。

我不知道它是否是由视频驱动程序和/或硬件引起的,但几乎可以肯定它是由干扰循环模式引起的。看起来很像应该是 GC 循环干扰了您的 for 循环(因此,随着内存被释放并可以分配,锯齿波突然下降)

【讨论】:

以上是关于锯齿 tkinter 主循环帧持续时间?的主要内容,如果未能解决你的问题,请参考以下文章

Python连载60-Tkinter布局按钮以及属性详解

Tkinter 嵌套主循环

为啥我的主循环在 tkinter 中不起作用?

Python D-Bus 和 Tkinter 主循环集成

Tkinter/Matplotlib 后端冲突导致无限主循环

Tkinter:在主循环中调用事件