1

目标: 30 分钟的定时活动,循环通过一系列交替缓慢和剧烈的锻炼。该窗口首先显示从 30 分钟开始的倒数计时器。练习以 30 秒的“慢”活动开始 - 计时器下方显示“慢”字样。然后继续进行 60 秒的剧烈运动。然后是 30 秒的慢速运动,然后是 60 秒的不同强度运动,以此类推。总共有 4 种不同的剧烈运动,每种运动都以缓慢的速度交替进行。

我已经组装了下面的代码。我不相信它是直观的。我不喜欢在方法周围蹦蹦跳跳,我觉得那里有一些方法可以用来提高效率。在实际层面上,这并不能很好地工作,因为计时器和消息(即显示的慢/运动#)开始变得不同步。我研究了 python 博客并不断“听到”关于 Tkinter 不是多线程的一些信息。我猜测 after(x,y) 会因为倒计时线程占用资源而受到损害......不太确定。另外,我想在每个方法之后添加一个音调作为“运动员”的音频队列无论如何,提前感谢您查看我的粗略代码。

 import Tkinter as tk

 class ExampleApp(tk.Tk):
     def __init__(self):
         tk.Tk.__init__(self)
         self.label = tk.Label(self, text="", width=10, font= ("Helvetica",72), background='yellow', fg = 'red')
         self.label2 = tk.Label(self, text="", width=10, font=("Helvetica",72), background='yellow', fg = 'blue')
         self.label.pack()
         self.label2.pack()
         self.remaining = 0
         self.countdown(1800)  # the 30 minutes countdown  initialization
         self.run = True 

     def countdown(self, remaining = None):
         if self.remaining == 1799:  # marker that starts the methods series
             self.start()
         if remaining is not None:
             self.remaining = remaining
         if self.remaining >= 1:
             mins = int(self.remaining/60)
             #rsecs = self.remaining - mins * 60 #could use this
             rsecs = self.remaining % 60     #same results as t - mins * 60
             self.remaining = self.remaining -1
             self.after(1000, self.countdown)
             self.label.configure(text ="%.2f" % (mins + rsecs / 100.0))
         else:
             self.label.configure(text="Stop")

     def start(self):
         app.after(0, self.slow1)
         return

     def slow1(self):
         self.label2.configure(text="Slow" )
         app.after(30000, self.fast1)
         return

     def fast1(self):
         self.label2.configure(text="Exercise 1" )
         app.after(61000, self.slow2)
         return

     def slow2(self):
         self.label2.configure(text="Slow" )
         app.after(31000, self.fast2)
         return

     def fast2(self):
         self.label2.configure(text="Exercise 2" )
         app.after(61000, self.slow3)
         return

     def slow3(self):
         self.label2.configure(text="Slow" )
         app.after(30000, self.fast3)
         return

     def fast3(self):
         self.label2.configure(text="Exercise 3" )
         app.after(60000, self.slow4)
         return

     def slow4(self):
         self.label2.configure(text="Slow" )
         app.after(30000, self.fast4)
         return

     def fast4(self):
         self.label2.configure(text="Exercise 4" )
         app.after(60000, self.slow1)
         return

 if __name__ == "__main__":
     app = ExampleApp()
     app.title("Intense Workout")
     app.geometry('550x550+200+200')
     app.configure(background='yellow')
     app.mainloop()
4

1 回答 1

1

有很多方法可以解决这个问题,而且你不需要线程。这是一个想法:

从定义间隔的数据结构开始。例如:

intervals = [[30, "slow"], [60, "intense"], [30, "slow"], [60, "intense]]

接下来,设置一个简单的计时器函数,该函数每秒调用一次。它每秒显示数字和列表头部的字符串。数字代表剩余时间。在该计时器函数中,从数字中减去 1。如果它变为零,则从列表中删除该元素。让这个函数在一秒钟后使用 调用自己after,当数据结构变空时停止(即:当只剩下一对并且数字下降到零时)。

以下为您提供有关如何实现此功能的想法。代码可以做得更好,但重点是说明如何使用 迭代数据结构after,而不一定要给出生产质量的示例。

import Tkinter as tk

class ExampleApp(tk.Tk):
    def __init__(self):
        tk.Tk.__init__(self)
        self.label = tk.Label(self, text="", width=10, font= ("Helvetica",72), background='yellow', fg = 'red')
        self.label2 = tk.Label(self, text="", width=10, font=("Helvetica",72), background='yellow', fg = 'blue')
        self.label.pack()
        self.label2.pack()
        self.intervals = [[30, "Slow"], [60, "Exercise 1"],
                          [30, "Slow"], [60, "Exercise 2"],
                          [30, "Slow"], [60, "Exercise 3"],
                          [30, "Slow"], [60, "Exercise 4"],
                          ]
        self.countdown()

    def countdown(self):
        (remaining, label) = self.intervals[0]
        self.label.configure(text=remaining)
        self.label2.configure(text=label)
        remaining -= 1
        if remaining < 0:
            self.intervals.pop(0)
        else:
            self.intervals[0][0] = remaining
        if len(self.intervals) > 0:
            self.after(1000, self.countdown)
        else:
            self.label.configure(text="Done!")

if __name__ == "__main__":
    app = ExampleApp()
    app.title("Intense Workout")
    app.geometry('550x550+200+200')
    app.configure(background='yellow')
    app.mainloop()
于 2012-08-15T12:50:11.757 回答