5

我最近开始学习 Python,我正在制作的简单应用程序的一部分包括一个计时器,该计时器具有在其自己的线程中运行的 hh:mm:ss 显示。

环顾网络,我发现了两种实现方式:

  1. 使用 sched.scheduler
  2. 使用 threading.Timer

对于两种实现,我的做法看起来都相似:

预定:

def tick(self, display, alarm_time):

    # Schedule this function to run every minute
    s = sched.scheduler(time.time, time.sleep)
    s.enter(1, 1, self.tick, ([display, alarm_time]))

    # Update the time
    self.updateTime(display)

定时器:

def tick(self, display):

    # Schedule this function to run every second
    t = Timer(1, self.tick, (display,alarm_time))
    t.start()

    # Update the time
    self.updateTime(display)
  1. 在正确滴答方面工作正常,但几分钟后会产生以下错误:RuntimeError: maximum recursion depth exceeded。我知道您可以手动增加最大递归级别,但是这里肯定不需要这样做吗?

  2. 没有错误,但偶尔会跳过秒,或不规则地滴答作响。

有人可以为我指出正确的方向吗?如何正确地做到这一点?谢谢你。

4

2 回答 2

5

以下是如何将一次性事件变成周期性事件,例如sched:如果函数必须创建自己的调度程序并且是唯一在其线程上运行的东西:

def tick(self, display, alarm_time, scheduler=None):
  # make a new scheduler only once & schedule this function immediately
  if scheduler is None:
    scheduler = sched.scheduler(time.time, time.sleep)
    scheduler.enter(0, 1, self.tick, ([display, alarm_time, scheduler]))
    scheduler.run()

  # reschedule this function to run again in a minute
  scheduler.enter(1, 1, self.tick, (display, alarm_time, scheduler]))

  # do whatever actual work this function requires, e.g.:
  self.updateTime(display)

如果还必须在同一线程中安排其他事件,则必须在“其他地方”制作和拥有调度程序——if上面的部分可以重构为另一种方法,例如:

def scheduleperiodic(self, method, *args):
  self.scheduler = sched.scheduler(time.time, time.sleep)
  self.scheduler.enter(0, 1, method, args)
  # whatever else needs to be scheduled at start, if any, can go here
  self.scheduler.run()

def tick(self, display, alarm_time):
  # reschedule this function to run again in a minute
  self.scheduler.enter(60, 1, self.tick, (display, alarm_time))

  # do whatever actual work this function requires, e.g.:
  self.updateTime(display)

当然,和往常一样sched,当调度程序运行时,它(和调度的事件回调)将“接管”有问题的线程(因此,如果您需要其他线程,则需要为它分离一个单独的线程同时发生的事情)。

如果您需要在许多函数中使用这种惯用语,可以将其重构为装饰器,但这会在某种程度上掩盖惯用语的潜在简单性,因此我更喜欢这种简单、公开的使用方式。顺便说一句,请注意 time.time 和 time.sleep 使用秒而不是分钟作为时间单位,因此您需要 60 而不是 1 来表示“从现在开始一分钟”;-)。

于 2009-09-10T15:57:34.487 回答
4

Timer 是一次性事件。不能以这种方式循环。

使用 Timer 调用一个函数,然后创建另一个 Timer,该 Timer 调用一个创建 Timer 的函数,该函数调用一个创建 Timer 的函数,...,必须达到递归限制。

您没有提及您的操作系统,但“跳过”或“不规则滴答”有两个原因。

  1. 您的计算机很忙,“1 秒”表示“非常接近 1 秒,具体取决于其他情况”

  2. 如果您在 0.9999 秒处启动计时器并等待 1 秒,则您可能处于 1.9999(向下舍入为 1)或 2.00000。它可能会出现重复时间或跳过时间。您计算机的内部硬件时钟非常准确,四舍五入到最接近的秒将(总是)导致重复或跳过的远程可能性。

正确使用sched。 http://docs.python.org/library/sched.html#module-sched

您的代码片段对 sched 也没有任何意义。您不需要创建新的调度程序对象。您只需要创建一个新事件

阅读http://docs.python.org/library/sched.html#sched.scheduler.enter了解为现有调度程序实例创建新事件。

于 2009-09-10T10:58:35.657 回答