iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >基于Python实现简单的定时器详解
  • 326
分享到

基于Python实现简单的定时器详解

2024-04-02 19:04:59 326人浏览 薄情痞子

Python 官方文档:入门教程 => 点击学习

摘要

所谓定时器,是指间隔特定时间执行特定任务的机制。几乎所有的编程语言,都有定时器的实现。比如,Java有util.Timer和util.TimerTask,javascript有set

所谓定时器,是指间隔特定时间执行特定任务的机制。几乎所有的编程语言,都有定时器的实现。比如,Java有util.Timer和util.TimerTask,javascript有setInterval和setTimeout,可以实现非常复杂的定时任务处理。然而,牛叉到无所不能的python,却没有一个像样的定时器,实在令人难以理解。

入门的同学一定会说:不是有个time.sleep吗?定好闹钟睡大觉,闹钟一响,起来干活,这不就是一个定时器吗?没错,time.sleep具备定时器的基本要素,但若作为定时器使用,则有两个致命的缺陷:一是阻塞主线程,睡觉的时候不能做任何事情;二是醒来以后需要主线程执行定时任务——即便使用线程技术,也得先由主线程来创建子线程。

说到这里,熟悉线程模块threading的同学也许会说:threading.Timer就是以线程方式运行的呀,既不会阻塞主线程,执行定时任务也无需主线程干预,这不就是一个完美的定时器吗?

我们先来看看threading.Timer是如何工作的。下面这段代码演示了threading.Timer的基本用法:启动定时器2秒钟后以线程方式调用函数do_something,在定时器等待的2秒钟内,以及do_something运行期间,主线程仍然可以做其他工作——此处是从键盘读取输入,借以阻塞主线程,以便观察定时器的工作情况。


import time
import threading

def do_something(name, gender='male'):
    print(time.time(), '定时时间到,执行特定任务' )
    print('name:%s, gender:%s'%(name, gender))

timer = threading.Timer(2, do_something, args=('Alice',), kwargs={'gender':'female'})
timer.start()
print(time.time(), '定时开始时间')
input('按回车键结束\n') # 此处阻塞住进程

正如我们所期待的那样,定时器启动2秒钟后,函数do_something被调用,这期间可以随时敲击回车键结束程序。这段代码的运行结果如下。

1627438957.4297626 定时开始时间

按回车键结束

1627438959.4299397 定时时间到,执行特定任务

name:Alice, gender:female

从使用效果看,threading.Timer称得上是一款简洁易用的定时器。不过,threading.Timer存在明显的短板,那就是不支持连续的定时任务,比如,每隔2秒钟调用一次do_something函数。如果一定要用threading.Timer实现连续定时,只能用类似嵌套的变通方法,在do_something函数中再次启动定时器。


import time
import threading

def do_something(name, gender='male'):
    global timer
    timer = threading.Timer(2, do_something, args=(name,), kwargs={'gender':gender})
    timer.start()
    
    print(time.time(), '定时时间到,执行特定任务' )
    print('name:%s, gender:%s'%(name, gender))
    time.sleep(5)
    print(time.time(), '完成特定任务' )

timer = threading.Timer(2, do_something, args=('Alice',), kwargs={'gender':'female'})
timer.start()
input('按回车键结束\n') # 此处阻塞住进程

这段代码重新定义了do_something函数,在函数开始位置启动下一次的定时任务。之所以放在开始位置,是为了保证两次定时之间的时间间隔尽可能精确。饶是如此,下面的运行结果显示,两次定时之间的时间间隔比设计的2秒钟多了大约10毫秒,且误差是连续累计的,重复执行100次,误差将会超过1秒钟。

按回车键结束

1627440628.683803 定时时间到,执行特定任务

name:Alice, gender:female

1627440630.6929214 定时时间到,执行特定任务

name:Alice, gender:female

1627440632.707388 定时时间到,执行特定任务

name:Alice, gender:female

1627440633.6890671 完成特定任务

1627440634.722474 定时时间到,执行特定任务

name:Alice, gender:female

1627440635.7092102 完成特定任务

1627440636.7277966 定时时间到,执行特定任务

name:Alice, gender:female

针对连续的定时任务,threading.Timer的表现还算差强人意,只是这种嵌套的写法完全颠覆了代码美学。对于像我这样有代码洁癖的程序员来说,是无法容忍和不可接受的。在我看来,一个完美的定时器应该满足以下5个条件,具备下图所示的结构。

  1. 不阻塞主线程
  2. 同时支持单次定时和连续定时
  3. 以线程或进程方式执行定时任务
  4. 定时任务的线程或进程的创建、运行,不影响定时精度
  5. 足够精确的定时精度,且误差不会累计

既然Python没有提供一个像样的定时器,那就自己写一个吧。下面这个定时器,满足上面提到的5个条件,最短时间间隔可以低至10毫秒,且误差不会累计。虽然还不够完美,但无论结构还是精度,都还说得过去。


import time
import threading

class PyTimer:
    """定时器类"""
    
    def __init__(self, func, *args, **kwargs):
        """构造函数"""
        
        self.func = func
        self.args = args
        self.kwargs = kwargs
        self.running = False
    
    def _run_func(self):
        """运行定时事件函数"""
        
        th = threading.Thread(target=self.func, args=self.args, kwargs=self.kwargs)
        th.setDaemon(True)
        th.start()
    
    def _start(self, interval, once):
        """启动定时器的线程函数"""
        
        if interval < 0.010:
            interval = 0.010
        
        if interval < 0.050:
            dt = interval/10
        else:
            dt = 0.005
        
        if once:
            deadline = time.time() + interval
            while time.time() < deadline:
                time.sleep(dt)
            
            # 定时时间到,调用定时事件函数
            self._run_func()
        else:
            self.running = True
            deadline = time.time() + interval
            while self.running:
                while time.time() < deadline:
                    time.sleep(dt)
                
                # 更新下一次定时时间
                deadline += interval
                
                # 定时时间到,调用定时事件函数
                if self.running:
                    self._run_func()
    
    def start(self, interval, once=False):
        """启动定时器
        
        interval    - 定时间隔,浮点型,以秒为单位,最高精度10毫秒
        once        - 是否仅启动一次,默认是连续的
        """
        
        th = threading.Thread(target=self._start, args=(interval, once))
        th.setDaemon(True)
        th.start()
    
    def stop(self):
        """停止定时器"""
        
        self.running = False

定时器类PyTimer实例化时,需要传入定时任务函数。如果定时任务函数有参数,也可以按照位置参数、关键字参数的顺序一并提供。PyTimer定时器提供start和stop两个方法,用于启动和停止定时器。其中stop方法不需要参数,start则需要一个以秒为单位的定时间隔参数。start还有一个布尔型的默认参数once,可以设置是否单次定时。once参数的默认值为False,即默认连续定时;如果需要单次定时,只需要将once置为true即可。


def do_something(name, gender='male'):
    print(time.time(), '定时时间到,执行特定任务' )
    print('name:%s, gender:%s'%(name, gender))
    time.sleep(5)
    print(time.time(), '完成特定任务' )

timer = PyTimer(do_something, 'Alice', gender='female')
timer.start(0.5, once=False)

input('按回车键结束\n') # 此处阻塞住进程
timer.stop()

上面是使用PyTimer定时器以0.5秒钟的间隔连续调用函数do_something的例子。这段代码的运行结果如下。

按回车键结束

1627450313.425347 定时时间到,执行特定任务

name:Alice, gender:female

1627450313.9226055 定时时间到,执行特定任务

name:Alice, gender:female

1627450314.421761 定时时间到,执行特定任务

name:Alice, gender:female

1627450314.9243422 定时时间到,执行特定任务

name:Alice, gender:female

1627450315.422722 定时时间到,执行特定任务

name:Alice, gender:female

1627450315.9200313 定时时间到,执行特定任务

name:Alice, gender:female

1627450316.4204514 定时时间到,执行特定任务

name:Alice, gender:female

1627450316.9215539 定时时间到,执行特定任务

name:Alice, gender:female

1627450317.4228196 定时时间到,执行特定任务

name:Alice, gender:female

1627450317.9245899 定时时间到,执行特定任务

name:Alice, gender:female

1627450318.42355 定时时间到,执行特定任务

name:Alice, gender:female

1627450318.4393418 完成特定任务

1627450318.9251466 定时时间到,执行特定任务

name:Alice, gender:female

1627450318.9395308 完成特定任务

1627450319.4242043 完成特定任务

1627450319.4242043 定时时间到,执行特定任务

name:Alice, gender:female

1627450319.9253905 定时时间到,执行特定任务

name:Alice, gender:female

1627450319.9411068 完成特定任务

1627450320.425871 完成特定任务

1627450320.425871 定时时间到,执行特定任务

name:Alice, gender:female

虽然每个定时任务需要运行5秒钟,但每隔0.5秒都会准时启动一个新的线程运行定时任务。从记录可以看出,尽管每次定时任务的启动时间有几个毫秒的误差,但误差不会累计,重复执行的时间间隔均值始终稳定在0.5秒。

 到此这篇关于基于Python实现简单的定时器详解的文章就介绍到这了,更多相关Python定时器内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 基于Python实现简单的定时器详解

本文链接: https://www.lsjlt.com/news/160465.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 基于Python实现简单的定时器详解
    所谓定时器,是指间隔特定时间执行特定任务的机制。几乎所有的编程语言,都有定时器的实现。比如,Java有util.Timer和util.TimerTask,JavaScript有set...
    99+
    2024-04-02
  • 基于Python怎样实现简单的定时器
    基于Python怎样实现简单的定时器,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。所谓定时器,是指间隔特定时间执行特定任务的机制。几乎所有的编程语言,都有定时器...
    99+
    2023-06-22
  • 基于Vue方法实现简单计时器
    Vue简单的计时器,供大家参考,具体内容如下 原理:setInterval来每隔1s(可设置的时间间隔)运行一次自增方法,clearInterval来让持续运行的自增方法停止,来达到...
    99+
    2024-04-02
  • 基于python的简单HTTP服务器实现
    HTTP协议 请求报文 请求头部字段解析 响应报文 响应头部字段解析 响应状态码 HTTP服务器实现 http协议大概是我们接触的最多的协议了,每打开一个网页,浏览器和服务器之间,使用的就是HTTP协议。...
    99+
    2023-01-31
    简单 服务器 python
  • Java简单实现定时器
    本文实例为大家分享了Java简单实现定时器的具体代码,供大家参考,具体内容如下 一、定时器 定时器相当于一个任务管理器。有些任务可能现在执行, 有些任务可能过1个小时,甚至很久才会执...
    99+
    2024-04-02
  • 基于Java快速实现一个简单版的HashMap详解
    目录1.示例图2.分析需求3.代码实现3.运行结果简单实现一个底层数据结构为数组 + 链表的HashMap,不考虑链表长度超过8个时变为红黑树的情况。 1.示例图 2.分析需求 p...
    99+
    2023-02-08
    Java实现HashMap Java HashMap
  • 基于OpenCV和Gradio实现简单的人脸识别详解
    目录环境配置实验原理程序设计1.上传图片2.人脸识别3.绘制方框4.保存预测结果5.最后运行gradio的lauch完整代码环境配置 gradio 安装 pip install gr...
    99+
    2023-05-14
    OpenCV Gradio实现人脸识别 OpenCV Gradio人脸识别 OpenCV Gradio
  • 基于python tkinter的简单计
    import tkinter #定义计算器类 class Calc: #初始化魔术方法 def __init__(self): #初始化共用属性 #定义一个用于存放被计算字符串的列表 self.operationLi...
    99+
    2023-01-31
    简单 python tkinter
  • 基于Python实现简单的学生点名系统
    目录本文背景效果展示实现方式1、读取excel表格2、搭建系统界面3、随机选择学生4、将代码打包成exe使用说明今年教师节前夕,我特意用Python做了个学生点名系统,非常好用,送给...
    99+
    2024-04-02
  • 基于Python实现简单的人脸识别系统
    目录前言基本原理代码实现创建虚拟环境安装必要的库前言 最近又多了不少朋友关注,先在这里谢谢大家。关注我的朋友大多数都是大学生,而且我简单看了一下,低年级的大学生居多,大多数都是为了完...
    99+
    2024-04-02
  • SpringBoot简单实现定时器过程
    目录前言正文定时器的实现cron表达式详解总结前言 公司最近需要完成一个定时去DB查询数据,把有问题的数据给筛选出来通过微信小程序的消息推送功能将有问题的数据推送给指定的员工去筛查问...
    99+
    2023-05-16
    Springboot定时器 Springboot定时器实现
  • 基于Java实现简单的时序数据压缩算法
    目录背景整数VarintZigZagDelta2Simple8b浮点数有损压缩无损压缩字符串标签型非标签型背景 今年在公司内部主导了两个的行情数据系统的构建,两者均使用到了常见的时序...
    99+
    2024-04-02
  • 基于Django的Admin后台实现定制简单监控页
    我们使用Django的Admin二次定制一个图形化界面,首先我们把语言设置为中文简体. 修改: settings.py LANGUAGE_CODE = 'zh-hans' TIM...
    99+
    2024-04-02
  • Java实现一个简单的定时器代码解析
    定时的功能我们在手机上见得比较多,比如定时清理垃圾,闹钟,等等.定时功能在java中主要使用的就是Timer对象,他在内部使用的就是多线程的技术.Time类主要负责完成定时计划任务的功能,就是在指定的时间的开始执行某个任务.Timer类的作...
    99+
    2023-05-30
    java 定时器 ava
  • 基于HTML+JS实现简单的年龄计算器
    目录前言演示效果HTML代码CSS代码Javascript代码演示地址前言 在线演示地址:http://haiyong.site/age-calculator JavaScript提...
    99+
    2024-04-02
  • 如何基于python实现单目三维重建详解
    目录一、单目三维重建概述二、实现过程(1)相机的标定(2)图像特征提取及匹配(3)三维重建三、结论四、代码总结一、单目三维重建概述 客观世界的物体是三维的,而我们用摄像机获取的图像是...
    99+
    2024-04-02
  • 基于springboot实现一个简单的aop实例
    简介 AOP(Aspect-Oriented Programming:面向切面编程) aop能将一些繁琐、重复、无关业务的逻辑封装起来,在一个地方进行统一处理,常用于日志记录、事务...
    99+
    2024-04-02
  • 关于Promise基本方法的简单实现
    目录前言catch() 方法done() 方法finally() 方法Promise.all() 方法Promise.race() 方法Promise.resolve() 和 Pro...
    99+
    2024-04-02
  • 基于Unity3D实现仿真时钟详解
    目录一、前言二、时钟小工具开发2-1、搭建UI2-2、实现脚本一、前言 今天实现一个时钟工具,其实在之前已经完成了一个简单的时钟工具:【Unity3D应用案例系列】时钟、钟表小组件开...
    99+
    2023-01-30
    Unity3D实现仿真时钟 Unity3D实现时钟 Unity3D时钟
  • 基于spring中的aop简单实例讲解
    aop,即面向切面编程,面向切面编程的目标就是分离关注点,比如:一个骑士只需要关注守护安全,或者远征,而骑士辉煌一生的事迹由谁来记录和歌颂呢,当然不会是自己了,这个完全可以由诗人去歌颂,比如当骑士出征的时候诗人可以去欢送,当骑士英勇牺牲的时...
    99+
    2023-05-31
    spring aop 实例
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作