设为首页 收藏本站
查看: 1017|回复: 0

[经验分享] Python任务调度之sched

[复制链接]

尚未签到

发表于 2018-8-4 10:10:16 | 显示全部楼层 |阅读模式
  这次我们主要讲解下Python自带模块当中的sched,不但小巧,也很强大,在实际应用中,某些场合还是可以用到的。作为一名Linux的SA,我们已经习惯了用crontab,而sched提供了一种延迟处理机制,也可以理解为任务调度的另一种方式的实现。
  scheduler.enter(delay, priority, action, argument)
  ● delay:延迟时间
  ● priority:优先级(数越大,优先越低)
  ● action:回调函数
  ● argument:回调函数的参数
  我们来写一个非常简单的例子:
  #!/usr/bin/env python
  # -*- coding: utf-8 -*-
  import sched
  import time
  scheduler = sched.scheduler(time.time, time.sleep)
  def func(name):
  print 'action: %s' % name , time.time()
  print 'START:', time.time()
  scheduler.enter(3, 2, func, ('fight',))
  print 'Middle'
  scheduler.enter(3, 1, func, ('make peace',))
  scheduler.run()
  print 'END:', time.time()
  time.sleep(5)
  运行结果如下:
  START: 1453357480.74
  Middle
  action: make peace 1453357483.74
  action: fight 1453357483.74
  END: 1453357483.74
  我们再举一个简单的例子说明下sched的其它特性:
  #!/usr/bin/env python
  # -*- coding: utf-8 -*-
  import sched
  import time
  scheduler = sched.scheduler(time.time, time.sleep)
  def func(name):
  print 'BEGIN: %s:' % name, time.time()
  time.sleep(2)
  print 'FINISH %s:' % name, time.time()
  print 'START:', time.time()
  scheduler.enter(2, 1, func, ('fight',))
  scheduler.enter(3, 1, func, ('make peace',))
  scheduler.run()
  print 'END:', time.time()
  time.sleep(20)
  运行结果如下:
  START: 1339665268.12
  BEGIN: fight: 1339665270.12
  FINISH fight: 1339665272.12
  BEGIN: make peace: 1339665272.12
  FINISH make peace: 1339665274.12
  END: 1339665274.12
  我们仔细观察下两次任务调度的时间间隔,发现是同时运行的?那又是为什么呢?run()一直被阻塞,直到所有事件被全部执行完. 每个事件在同一线程中运行,所以如果一个事件的执行时间大于其他事件的延迟时间,那么,就会产生重叠。重叠的解决方法是推迟后来事件的执行时间。这样保证 没有丢失任何事件,但这些事件的调用时刻会比原先设定的迟。
  上面的例子第二个事件在第一个事件运行结束后立即运行,因为第一个事件的执行时间足够长,已经超过第二个事件的预期开始时刻。(本来应该1339660903秒运行)
  我们再介绍另外一个保证action在同一时刻执行的函数:
  scheduler.enterabs(time, priority, action, argument)
  #!/usr/bin/env python
  # -*- coding: utf-8 -*-
  import sched
  import time
  scheduler = sched.scheduler(time.time, time.sleep)
  now = time.time()
  def func(name):
  print 'action:', time.time(), name
  print 'START:', now
  scheduler.enterabs(now + 2, 2, func, ('make peace',))
  scheduler.enterabs(now + 2, 1, func, ('fight',))
  scheduler.run()
  print 'END:', now
  time.sleep(20)
  运行结果如下:
  START: 1339666232.38
  action: 1339666234.38 fight
  action: 1339666234.38 make peace
  END: 1339666232.38
  因为优先级的关系,所以先fight,然后再make peace,打架是如此重要....总体来讲,如果想单纯的替换crontab的话,Scheduler框架更加适合,做延迟任务的调度处理的话sched还是可以考虑的。
  如果我们想要取消任务调度,可以使用cancel()函数。在上面的例子中出现了阻塞延迟的现象,如果引用线程机制就会避免这种情况的发生,我们简单举个例子:
  #!/usr/bin/env python
  # -*- coding: utf-8 -*-
  import sched
  import threading
  import time
  scheduler = sched.scheduler(time.time, time.sleep)
  counter = 0
  def increment_counter(name):
  global counter
  print 'action: %s' % name , time.time()
  counter += 1
  print 'counter: ', counter
  print 'START:', time.time()
  action1 = scheduler.enter(2, 1, increment_counter, ('action1',))
  action2 = scheduler.enter(3, 1, increment_counter, ('action2',))
  t = threading.Thread(target=scheduler.run)
  t.start()
  scheduler.cancel(action1)
  t.join()
  print 'counter:', counter
  print 'END:', time.time()
  time.sleep(20)
  运行结果如下:
  START: 1339666987.27
  action: action2 1339666990.27
  counter:  1
  counter: 1
  END: 1339666990.27
  因为run()函数会引起阻塞,所以我们需要采用线程机制的方法在另一个线程中通过对象的引用取消任务调度,这里只调度了action2方法。
  本文出自 “放飞翅膀,追求梦想” 博客,请务必保留此出处http://flykite.blog.51cto.com/4721239/898649
  python使用sched模块周期性抓取网页内容
  1.使用sched模块可以周期性地执行指定函数
  2.在周期性执行指定函数中抓取指定网页,并解析出想要的网页内容,代码中是六维论坛的在线人数
  论坛在线人数统计
  #coding=utf-8
  import time,sched,os,urllib2,re,string
  #初始化sched模块的scheduler类
  #第一个参数是一个可以返回时间戳的函数,第二个参数可以在定时未到达之前阻塞。
  s = sched.scheduler(time.time,time.sleep)
  #被周期性调度触发的函数
  def event_func():
  req = urllib2.Request('http://bt.neu6.edu.cn/')
  response = urllib2.urlopen(req)
  rawdata = response.read()
  response.close()
  usernump = re.compile(r'总计 <em>.*?</em> 人在线')
  usernummatch = usernump.findall(rawdata)
  if usernummatch:
  currentnum=usernummatch[0]
  currentnum=currentnum[string.index(currentnum,'>')+1:string.rindex(currentnum,'<')]
  print "Current Time:",time.strftime('%Y,%m,%d,%H,%M',time.localtime(time.time())),'User num:',currentnum
  # 保存结果,供图表工具amcharts使用
  result=open('liuvUserNUm','a')
  result.write('{year: new Date('+time.strftime('%Y,%m,%d,%H,%M',time.localtime(time.time()))+'),value:'+currentnum+'},\n')
  result.close()
  #enter四个参数分别为:间隔事件、优先级(用于同时间到达的两个事件同时执行时定序)、被调用触发的函数,给他的参数(注意:一定要以tuple给如,如果只有一个参数就(xx,))
  def perform(inc):
  s.enter(inc,0,perform,(inc,))
  event_func()
  def mymain(inc=900):
  s.enter(0,0,perform,(inc,))
  s.run()
  if __name__ == "__main__":
  mymain()
  本文出自别人博客,请务必保留此出处http://outofmemory.cn/code-snippet/3199/python-usage-sched-module-period-xing-zhuaqu-wangye-content

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-546322-1-1.html 上篇帖子: linux下python 连接mysql 下篇帖子: Python socket.help Methods
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表