当前位置:首页 > 技术学院 > 技术前线
[导读]服务器编程中一块是定时器,影响着服务器性能 定时器一个作用是用于定时检测客户端连接,并踢掉非活动连接; 定时器一般会把定时事件封装成定时器,并进行组织以方便管理

计时

在linux中,一般由文件下的alarm函数和setitimer来设置定时器,到时间则发出SIGALARM,并调用指定的到期信号处理函数,

signal(SIGALRM, handler);函数来设置到期事件处理函数,而这个到期事件处理函数在游双的书例子里可以看到,是定时器类里的tick函数,即每隔一个周期调用tick()一次,注意要区分timer到期的回调函数和SIGALARM的回调函数

封装timer

在服务器编程里,要处理定时事件,会封装个定时器timer类,其中一般会包含到期时间和回调函数,如muduo中timer.h的实现,和l游双书中的定时器那一章的种种例子,并根据不同实现方式增加额外的数据如链表的节点

而定时器应该给出的用户接口有注册定时器,注销定时器;注册定时器应增加区分重复定时器和一次性定时器

还应该给出定期触发的函数,即上面提到的tick(),在tick函数里判断timer里哪些是已经过期的,触发定时事件,根据定时器是否是重复的删除或者重新设置此定时器

定时器实现类型

定时器会用到什么操作呢?它的插入,指定注销,注销到期的定时器,根据这几点看一下如何设计定时器,ps.注意区分指定注销和注销到期,因为以下的实现大多是已经排序好的,注销到期的一般是从头开始往下找,而指定注销是注销当中某个节点的定时器

先说明libevent中的实现是最小堆,muduo是采用了红黑树来实现,以下列出几种类型的定时器实现

最简单的实现:双向链表

直接利用链表实现,每一个定时器作为一个链表的节点,这样做最直观,而几种操作的复杂度是:

添加的时候就直接插入到链表末端,时间复杂度O(1)

找到到期timer,则需要遍历全部,时间复杂度为O(n)

代码请看参考资料[1]

优化的链表:排序双向链表

优化操作:每次插入按到期时间进行排序,时间复杂度是:

插入为O(n),找到到期timer时间复杂度是O(1)的操作,指定timer删除操作,是O(1)的复杂度,这也是为什么要用双向链表的原因,直接传入一个链表节点进行删除,,ps:这里排序链表判断到期虽然会有个while循环,是为了找到地一个非过期并执行前面过期的所有回调函数,平均下来还是个O(1)的操作

代码:参考资料[1]

最小堆实现

优化点:最小堆的插入操作是log(n),参考下堆的插入操作:插入到堆最后面以后,进行上浮调整,最大调整次数为树的高度,即log(n),

到期触发的时间复杂度为O(1),及取最值,而堆这个结构最适合做这个,在游双的书中能看到,最小堆的实现alarm信号发送的时间设置成堆中最近的触发时间,每次取完后其实还有个log(n)的heapify调整时间,估计参考资料[1]和游双都把这个操作延后到平时(即延迟销毁,游双的书第11章第215页)了,

,指定删除操作则略微麻烦,这也是为什么muduo不采用这个方法的原因

(如果del_timer函数的参数,传入timer作参数直接删除再堆化一下就行,如果传入参数为定时器序号,则遍历到再删除为O(n),用一个map来存序号到定时器在堆中位置,则为时间O(1))(但是每次变换都)

代码:参考资料[1]或者游双的书

红黑树实现

muduo的实现,顺便提下,heap比起红黑树的好处是,像陈硕书中说的:内存的局部性更好,参考资料[1]说的内存使用率更好,且性能会相差一点,

红黑树对比堆查找特定的timer速度会快一点(参考[1]说的)

有时间复杂度就不分析了,muduo书中还提到了了一下set,map,multiset,multimap的选择

代码:参考muduo(直接用了现成的数据结构)或者参考资料[1]或者游双的书

时间轮

时间轮的介绍先略过TODO在游双的书中和陈硕的书中示例代码部分都有提到,

前文没有分清定时器的概念,晚点再改TODO

首先要把socket fd和定时器的指针加以封装,以便删除使用,即可以通过这个类来访问fd和timer*,要删除时间轮中的某个fd对应的timer*我们直接访问这个数据结构,在游双的书里取名为client_data

封装timer类,包括时间,到期执行的回调函数,这个类是以便时间轮使用,在时间轮中,我们是直接用timer*来排序使用的,

分析下各种操作的时间复杂度

插入:直接调用API传入timer*,而时间轮里每个轮里的是链表,链表删除直接O(1)

添加:插入操作是对时间进行取模放入那个槽中,时间复杂度也是O(1)

删除:直接传入timer*,O(1)

到期触发:是n个定时器,p个槽,根据哈希函数不同值均匀分布的特点,大概时间复杂度是O(n/p),游双的书中称之为:比O(n)好很多

使用IO复用

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

项目开发为各种需要精确时间控制和频率测量的应用场景提供了关键支持。借助C语言在单片机编程中的强大表现力,我们能够充分发挥定时器计数器的功能,实现精准计时与频率测量,让单片机在众多领域大显身手。

关键字: 单片机 定时器

单片机程序开发,定时器计数器犹如一把精准的瑞士军刀,为各种时间相关和计数任务提供了强大而灵活的支持。无论是实现精确的定时控制、测量脉冲宽度,还是进行频率计数,定时器计数器都扮演着不可或缺的角色。对于初学者而言,掌握定时器...

关键字: 定时器 单片机

2025年12月4日,致力于亚太地区市场的国际领先半导体元器件分销商---大联大控股宣布,其旗下品佳推出基于英飞凌(Infineon)TLE9954芯片的电机控制器方案。

关键字: 电机控制器 驱动器 定时器

单片机定时器通过晶振频率来产生精准的计数脉冲,从而实现微秒级别的时间控制。 单片机定时器依托于一个稳定的计数器,该计数器与单片机上的晶振部件相连。晶振经过12分频后,为单片机提供稳定且精准的1MHZ脉冲。由于晶振的频率极...

关键字: 单片机 定时器

编码器‌是一种传感器,主要用于测量旋转运动的角度、角速度和位移等参数。常见的编码器类型包括增量式编码器和绝对值编码器,前者通过检测脉冲的增加或减少来测量旋转方向和距离,后者则直接输出当前的绝对位置信息‌。

关键字: 编码器‌ 定时器

一款语音控制的番茄时间定时器,内置Seeed Studio XIAO ESP32S3感测和圆形显示。用免提时间管理保持高效!

关键字: 定时器 ESP32S3 语音控制

【2025年3月20日, 德国慕尼黑讯】全球功率系统和物联网领域的半导体领导者英飞凌科技股份公司(FSE代码:IFX / OTCQX代码: IFNNY)近日宣布,其旗下ModusToolboxTM开发平台中的AIROC™...

关键字: MCU 物联网 定时器

众所周知,说到延时,很多人都会想到用软件件来实现,比如定时器之类的。今天就来说说用硬件来实现定时的方式,虽说没有那么准,但是有些场合还是用得到的。

关键字: CD4060 定时器
关闭