每隔一定时间执行一次函数,而不计算函数

时间:2016-07-13 08:39:10

标签: python timer scheduled-tasks

我需要在特定时间执行每隔一定时间间隔的函数。

例如,如果开始时间是00:00:00,间隔是5秒,那么我需要执行此功能的时间是:

00:00:00
00:00:05
00:00:10
00:00:15
...
23:59:55

你可以看到我的观点是系统的时钟

在stackoverflow中存在很多关于这个问题的问题 How to execute a function asynchronously every 60 seconds in Python?What is the best way to repeatedly execute a function every x seconds in Python?。此外,我看到了schedule 0.3.2的其他解决方案。

我的问题是,例如使用此代码:

import schedule
import time

def job():
    print("I'm working...")
    print 'execute f time: ', time.ctime(time.time())
    print 'execute f time: ', time.time()
schedule.every().minutes.do(job)

while 1:
    schedule.run_pending()
    time.sleep(1)

在这种情况下,当我运行我的代码时,我的代码的输出是:

I'm working...
execute f time:  Wed Jul 13 04:32:00 2016
execute f time:  1468398720.88
I'm working...
execute f time:  Wed Jul 13 04:33:00 2016
execute f time:  1468398780.94
I'm working...
execute f time:  Wed Jul 13 04:34:01 2016
execute f time:  1468398841.0

正如您所看到的,此代码每分钟执行一次,但间隔从1分钟变为1分钟1秒,此错误对我来说很重要,可能是打印功能时间的原因。我需要在这种情况下每隔几分钟继续执行,而不会由函数打印的累计时间总和引起的秒数

我的计时器解决方案有同样的错误:

import threading
import time

def f():
    # call f() again in 10 seconds
    threading.Timer(10, f).start()
    # do something here ...
    print 'execute f time: ', time.time()
    time.sleep(5)

# start calling f now and every 10 sec thereafter
print 'init time: ', time.time()
f()

输出的一个片段显示某个时刻的推理错误导致间隔大于5([0,5,10,16,21,26,....])这对我来说是错误的,因为对我来说,执行函数的时间非常重要:

init time:  1468407868.6
execute f time:  1468407868.6
execute f time:  1468407878.6
execute f time:  1468407888.6
execute f time:  1468407898.6
execute f time:  1468407908.61  # change the interval
execute f time:  1468407918.61
execute f time:  1468407928.61

随着模块sched我有同样的问题

请有任何建议 提前谢谢。

更新

好的我测试crontab并且不满足我。首先我在Ubuntu中对我的用户的配置crontab文件进行测试,但后来我找到了一个库来编写crontab文件配置的指令,所以我使用了库Crontab

#!/usr/bin/python
# -*- coding: utf-8 -*-

from crontab import CronTab
task = CronTab(user='cyberguille')
command='/home/cyberguille/date.py'
cron_job=task.new(command)
cron_job.setall('*/1, *, *, *, *')
task.write()
print task.render()

正如你所看到的每一分钟执行代码,但我看到两个问题,在不到一分钟内执行并不容易(见How to run scripts every 5 seconds using cron?)我不知道为什么每执行一分钟就是每一分钟一秒钟,我们有太累积的总和错误,但错误比上面的例子更合适

date.py中的代码是这个

#!/usr/bin/python
# -*- coding: utf-8 -*-

#import time
from datetime import datetime
tiempo = datetime.now()
#tiempo = time.strftime("%H:%M:%S")
archivo=open('/home/cyberguille/archivo_fecha.txt', 'a+')

archivo.write(str(tiempo) + '\n')

并且输出的片段是

2016-07-19 00:11:01.307779
2016-07-19 00:12:01.365995
2016-07-19 00:13:01.421373
2016-07-19 00:14:01.477752
2016-07-19 00:15:01.532234
2016-07-19 00:16:01.588818
2016-07-19 00:17:01.649581
2016-07-19 00:18:01.705975
2016-07-19 00:19:01.759986
2016-07-19 00:20:01.816754
2016-07-19 00:21:01.873748
2016-07-19 00:22:01.927750
2016-07-19 00:23:01.980666
2016-07-19 00:24:02.036164
2016-07-19 00:25:01.090912
2016-07-19 00:26:01.148098

真的每隔一分钟就可以说这不是重要错误,但为什么不能执行例如00中的秒数

2016-07-19 00:11:00.307779
2016-07-19 00:12:00.365995
2016-07-19 00:13:00.421373
2016-07-19 00:14:00.477752

也许调用脚本很慢。

我找到了一个效果很好的解决方案,这是脚本

    from multiprocessing import Process
    from datetime import datetime
    import time
    import sched

    def timer(interval):
            sc = sched.scheduler(time.time, time.sleep)
            sc.enter(1, 1, do_something, (sc,interval))
            sc.run()
     def do_something(sc,interval):
            dateTime = datetime.now()
            if(dateTime.second%interval==0):
                p = Process(target=SensorContainer.run,args=(dateTime,))
                p.start()
            sc.enter(1, 1, SensorContainer.do_something, (sc,interval,))


     def run(datetime):
            print datetime

使用这个解决方案,计时器在0秒内每隔一分钟工作一次,因为你可以看到我的测量是在我调用函数的时候,这与crontab中的不同,但无论如何我测试在run函数中花时间a也是秒秒。

1 个答案:

答案 0 :(得分:3)

我相信你最好依靠操作系统进行调度而不是Python。使用单个函数创建一个简单的脚本然后使用crontab(如果使用Linux)或Windows任务计划程序来安排执行将更简单。

优选OS方法的一个原因是稳健性。例如,如果您依赖Python进行调度,以防发生未处理的异常,那么您的脚本将终止,直到它再次手动执行。同样适用于系统重启等。

如果您正在使用操作系统进行调度,即使在系统重新启动和未处理的异常情况下,脚本仍将继续执行预定义的计划。