10 def __init__(self, interval, function, args=None, kwargs=None): # 初始化的时候传参是延迟时间、调用的函数,函数的可变位置参数、函数的可变关键字参数 11 Thread.__init__(self) # 调用Thread类初始化配置实例 12 self.interval = interval # 在使用Thread类初始化配置实例之后再额外的增加interval属性 13 self.function = function # 同理再额外的增加function属性 14 self.args = args if args is not None else [] # 如果args不是空的话就使用args,如果是空就给一个空list 15 self.kwargs = kwargs if kwargs is not None else {} # 同理,kwargs不是空的就是kwargs,如果是空就给一个空字典

threading.Timer的作用

官方给的定义是:

"""Call a function after a specified number of seconds:

        t = Timer(30.0, f, args=None, kwargs=None)
        t.start()
        t.cancel()     # stop the timer's action if it's still waiting

"""

意思是:

  在一个特定的秒数之后调用一个函数,使用方法是创建一个Timer实例,然后start()启动线程,如果在线程调用传入的函数之前可以使用cancel进行取消。   

threading.Timer源码分析    

1 class Timer(Thread):  # 继承自Thread类
 2     """Call a function after a specified number of seconds:
 3 
 4             t = Timer(30.0, f, args=None, kwargs=None)
 5             t.start()
 6             t.cancel()     # stop the timer's action if it's still waiting
 7 
 8     """   
 9 
10     def __init__(self, interval, function, args=None, kwargs=None):  # 初始化的时候传参是延迟时间、调用的函数,函数的可变位置参数、函数的可变关键字参数
11         Thread.__init__(self)  # 调用Thread类初始化配置实例
12         self.interval = interval  # 在使用Thread类初始化配置实例之后再额外的增加interval属性
13         self.function = function  # 同理再额外的增加function属性
14         self.args = args if args is not None else []   # 如果args不是空的话就使用args,如果是空就给一个空list
15         self.kwargs = kwargs if kwargs is not None else {}  # 同理,kwargs不是空的就是kwargs,如果是空就给一个空字典
16         self.finished = Event()  # 再添加一个属性finished,是一个Event类的实例,这里知道Event类的实例用法就知道它在这里要怎么用了
17 
18     def cancel(self):
       """如果finished属性还没有被set,即函数function还没有被调用的之前阻止,因为下面函数调用之前会判断finished是否被set了,所以这里赶在调用之前注定set就能够阻止后面的调用。"""
19         """Stop the timer if it hasn't finished yet."""
20         self.finished.set()
21 
22     def run(self):  # 继承自Thread类,并且重写了Thread类,我们分析Thread类的源码会发现,start()方法会主动调用self.run(),
              # 我们Timer类没有实现start()方法,这样Timer类实例在执行start()的时候会跑到父类Thread上,然后调用父类的start,
              # 在父类的start()方法中会有一句self.run()来调用工作线程中的函数,这里self是Timer的实例,所以,这里可以重写run就可以设定run的时间了。
23         self.finished.wait(self.interval)  # 这里使用Event类的实例的wait方法,等待了我们设定的self.interval时间,然后关键点是下面一句
24         if not self.finished.is_set():  # 这一句是关键点,检查一下是否被set了,如果没有被set了就调用传入的函数,如果被set了有两种情况:
                          # 第一种情况是在self.finished.wait(self.interval)的期间,我们调用cancel主动提前set了;
                                           # 第二种情况是已经start()过一次了,这里就不能再进行start了这样就和父类的保持了一致:即一个线程只能够start一次
25             self.function(*self.args, **self.kwargs)
26         self.finished.set()  # 调用完成后set,即便之前已经被set了,这里还可以被set,因为Event实例可以被set多次。