我需要在特定时间执行每隔一定时间间隔的函数。
例如,如果开始时间是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也是秒秒。
答案 0 :(得分:3)
我相信你最好依靠操作系统进行调度而不是Python。使用单个函数创建一个简单的脚本然后使用crontab(如果使用Linux)或Windows任务计划程序来安排执行将更简单。
优选OS方法的一个原因是稳健性。例如,如果您依赖Python进行调度,以防发生未处理的异常,那么您的脚本将终止,直到它再次手动执行。同样适用于系统重启等。
如果您正在使用操作系统进行调度,即使在系统重新启动和未处理的异常情况下,脚本仍将继续执行预定义的计划。