chzhiyi / -KnowledgeShare

6 stars 1 forks source link

20190415 - Python定时器的基本使用 - jingchu #52

Open qingfengmingjian opened 5 years ago

qingfengmingjian commented 5 years ago

在实际应用中,我们经常需要使用定时器去触发一些事件。

Python中通过线程实现定时器timer, 其使用非常简单。

import threading
import time

def fun_timer():
    print('Hello Timer!')

while True:  #通过while来循环执行定时器
timer = threading.Timer(5, fun_timer)#一秒后,执行后面的函数
timer.start()
time.sleep(1)#如果不设置暂停时间,那么除了第一次是5秒后执行,其他是马上执行

一段时间后调用一个函数,但并没有说要循环调用该函数。因此,修改如下:

def fun_timer():
    print('Hello Timer!')
    global timer   #全局变量
    timer = threading.Timer(5.5, fun_timer)
    timer.start()
fun_timer()
timer = threading.Timer(1, fun_timer)#通过递归完成循环,间隔时间为5.5
timer.start()

1)定时器构造函数主要有2个参数,第一个参数为时间,第二个参数为函数名, 第一个参数表示多长时间后调用后面第二个参数指明的函数。 第二个参数注意是函数对象,进行参数传递,用函数名(如fun_timer)表示该对象, 不能写成函数执行语句fun_timer(),不然会报错。用type查看下,可以看出两者的区别。 '''

print(type(fun_timer()))#<class 'NoneType'>错误的用法 print(type(fun_timer))#<class 'function'>正确的用法

''' 2)必须在定时器执行函数内部重复构造定时器,因为定时器构造后只执行1次,必须循环调用。

3)定时器间隔单位是秒,可以是浮点数,如5.5,0.02等,在执行函数fun_timer内部和外部中给的值可以不同。 如上例中第一次执行fun_timer是1秒后,后面的都是5.5秒后执行。


可以使用cancel停止定时器的工作,如下例:

def fun_timer():
    print('Hello Timer!')
    global timer
    timer = threading.Timer(5.5, fun_timer)
    timer.start()

timer = threading.Timer(1, fun_timer)
timer.start()
time.sleep(15)  # timer.start()和time.sleep同时运行,15秒后执行timer.cancel()
timer.cancel() #停止定时器,是对主线程的取消,有兴趣可以了解定时器的循环是启用单个线程还是多个线程

下面是一个Python写的定时器,定时精度可调节:
import sys
import os
import getopt
import threading
import time

def Usage():
    usage_str = '''说明:
    \t定时器
    \timer.py -h 显示本帮助信息,也可以使用--help选项
    \timer.py -d num 指定一个延时时间(以毫秒为单位)
    \t          也可以使用--duration=num选项
    '''
    print(usage_str)

def args_proc(argv):
    '''处理命令行参数'''
    try:
        opts, args = getopt.getopt(sys.argv[1:], 'hd:', ['help', 'duration='])
    except getopt.GetoptError as err:
        print('错误!请为脚本指定正确的命令行参数。\n')
        Usage()
        sys.exit(255)
    if len(opts) < 1:
        print('使用提示:缺少必须的参数。')
        Usage()
        sys.exit(255)
    usr_argvs = {}
    for op, value in opts:
        if op in ('-h', '--help'):
            Usage()
            sys.exit(1)
        elif op in ('-d', '--duration'):
            if int(value) <= 0:
                print('错误!指定的参数值%s无效。\n' % (value))
                Usage()
                sys.exit(2)
            else:
                usr_argvs['-d'] = int(value)
        else:
            print('unhandled option')
            sys.exit(3)
    return usr_argvs

def timer_proc(interval_in_millisecond):
    loop_interval = 10  # 定时精度,也是循环间隔时间(毫秒),也是输出信息刷新间隔时间,它不能大于指定的最大延时时间,否则可能导致无任何输出
    t = interval_in_millisecond / loop_interval
    while t >= 0:
        min = (t * loop_interval) / 1000 / 60
        sec = (t * loop_interval) / 1000 % 60
        millisecond = (t * loop_interval) % 1000
        print('\rThe remaining time:%02d:%02d:%03d...' % (min, sec, millisecond), end='\t\t')
        time.sleep(loop_interval / 1000)
        t -= 1
    if millisecond != 0:
        millisecond = 0
        print('\rThe remaining time:%02d:%02d:%03d...' % (min, sec, millisecond), end='\t\t')
    print()

#### 应用程序入口
if __name__ == '__main__':
    usr_argvs = {}
    usr_argvs = args_proc(sys.argv)
    for argv in usr_argvs:
        if argv in ('-d', '--duration'):
            timer_proc(usr_argvs[argv])
        else:
            continue