博客
关于我
定时器的实现
阅读量:730 次
发布时间:2019-03-21

本文共 2188 字,大约阅读时间需要 7 分钟。

定时器设计概述

1. 定时器概述

定时器是服务端应用程序中负责执行时间相关任务的核心组件。服务端逻辑主要由两个事件驱动:网络事件和时间事件。不同框架对这两种事件的处理方式有所不同。

在单线程模型中(如 Nginx、Redis 等),网络事件和时间事件通过同一线程同步处理;而在多线程模型中(如 Skynet 等),则分别由不同的线程处理网络事件和时间事件。

1.1 单线程定时器实现

while (!quit) {    int now = get_now_time(); // 单位:ms    int timeout = get_nearest_timer() - now;    if (timeout < 0) timeout = 0;    int nevent = epoll_wait(epfd, ev, nev, timeout); // 利用 epoll_wait 实现定时器    for (int i = 0; i < nevent; i++) {        // 处理网络事件    }}

1.2 多线程定时器实现

void* thread_timer(void* thread_param) {    init_timer();    while (!quit) {        update_timer(); // 更新定时器检测        sleep(t); // sleep 时间 t    }    clear_timer();    return NULL;}pthread_create(&pid, NULL, thread_timer, &thread_param);

2. 定时器设计

2.1 接轴设计

// 初始化定时器void init_timer();// 添加定时器cbNode* add_timer(int expire, callback cb);// 删除定时器bool del_timer(Node* node);// 找到最近要触发的定时任务Node* find_nearest_timer();// 更新定时器检测void update_timer();

2.2 数据结构选择

时间轮需要高效的数据结构支持,主要有以下四种选择:

2.2.1 红黑树

红黑树可以支持 O(logN) 增、删、查操作。 дополнитель有序树结构允许快速查找最小节点,但需要维护节点的有序性。

2.2.2 最小堆

最小堆以完全二叉树形式存在,支持 O(logN) 增、查操作,删除操作通过辅助数据结构加速。堆的最小节点总是根节点。

2.2.3 跳表

跳表的增、删、查操作复杂度为 O(logN),但其空间复杂度较高。虽然最小节点查找时间复杂度为 O(1),但大数据量下不具备高效性。

2.2.4 时间轮

时间轮可以在 O(1) 时间复杂度下完成增、删、查操作,最小节点查找也为 O(1)。适合多个定时任务同时触发的情况。

2.3 红黑树实现定时器

void ngx_rbtree_insert_timer_value(ngx_rbtree_node_t *temp, ngx_rbtree_node_t *node, ngx_rbtree_node_t *sentinel) {    ngx_rbtree_node_t **p;    for ( ;; ) {        p = ((ngx_rbtree_key_int_t)(node->key - temp->key) < 0) ? &temp->left : &temp->right;        if (*p == sentinel) {            break;        }        temp = *p;    }    *p = node;    node->parent = temp;    node->left = sentinel;    node->right = sentinel;    ngx_rbt_red(node);}

2.4 最小堆实现定时器

最小堆的增操作需要满足完全二叉树定义,加入新节点后可能需要上浮操作。删除操作需要借由上升或下沉来维护堆的结构。

2.5 时间轮实现定时器

int seconds[60]; // 表盘刻度描述int tick = 0;// 时间轮每秒移动一次while (!quit) {    tick = (tick + 1) % 60; // 秒针循环}

3. 时间轮应用实例

3.1 单层级时间轮

用于解决心跳检测中常见的时延问题。例如,客户端每5秒发送心跳包,服务端若10秒内未收到心跳包则清除连接。

单层级时间轮设计

  • 准备一个固定大小的数组存储连接数据。
  • 每秒检查一次心跳包发送情况。
  • 使用指针移动数组位置,实现心跳检测。
  • 多层级时间轮设计

    通过将定时任务按优先级分层,实现多级时间轮共享机制。例如,紧急任务放在第一层,普通任务放在第二层等。

    3.2 时间轮优化建议

  • 合理设置数组长度,考虑连接数量和心跳检测时间。
  • 优化指针移动方式,避免重复计算。
  • 维护使用计数器,确保心跳检测逻辑正确。
  • 通过以上设计,可以编写高效、灵活的定时器系统,满足不同场景的服务端需求。

    转载地址:http://xjngz.baihongyu.com/

    你可能感兴趣的文章
    NIFI大数据进阶_NIFI的模板和组的使用-介绍和实际操作_创建组_嵌套组_模板创建下载_导入---大数据之Nifi工作笔记0022
    查看>>
    NIFI大数据进阶_NIFI监控功能实际操作_Summary查看系统和处理器运行情况_viewDataProvenance查看_---大数据之Nifi工作笔记0026
    查看>>
    NIFI大数据进阶_NIFI监控的强大功能介绍_处理器面板_进程组面板_summary监控_data_provenance事件源---大数据之Nifi工作笔记0025
    查看>>
    NIFI大数据进阶_NIFI集群知识点_认识NIFI集群以及集群的组成部分---大数据之Nifi工作笔记0014
    查看>>
    NIFI大数据进阶_NIFI集群知识点_集群的断开_重连_退役_卸载_总结---大数据之Nifi工作笔记0018
    查看>>
    NIFI大数据进阶_使用NIFI表达式语言_来获取自定义属性中的数据_NIFI表达式使用体验---大数据之Nifi工作笔记0024
    查看>>
    NIFI大数据进阶_内嵌ZK模式集群1_搭建过程说明---大数据之Nifi工作笔记0015
    查看>>
    NIFI大数据进阶_内嵌ZK模式集群2_实际操作搭建NIFI内嵌模式集群---大数据之Nifi工作笔记0016
    查看>>
    NIFI大数据进阶_外部ZK模式集群1_实际操作搭建NIFI外部ZK模式集群---大数据之Nifi工作笔记0017
    查看>>
    NIFI大数据进阶_实时同步MySql的数据到Hive中去_可增量同步_实时监控MySql数据库变化_实际操作_03---大数据之Nifi工作笔记0035
    查看>>
    NIFI大数据进阶_实时同步MySql的数据到Hive中去_可增量同步_实时监控MySql数据库变化_操作方法说明_01---大数据之Nifi工作笔记0033
    查看>>
    NIFI大数据进阶_实时同步MySql的数据到Hive中去_可增量同步_实时监控MySql数据库变化_操作方法说明_02---大数据之Nifi工作笔记0034
    查看>>
    NIFI大数据进阶_离线同步MySql数据到HDFS_01_实际操作---大数据之Nifi工作笔记0029
    查看>>
    NIFI大数据进阶_离线同步MySql数据到HDFS_02_实际操作_splitjson处理器_puthdfs处理器_querydatabasetable处理器---大数据之Nifi工作笔记0030
    查看>>
    NIFI大数据进阶_离线同步MySql数据到HDFS_说明操作步骤---大数据之Nifi工作笔记0028
    查看>>
    NIFI大数据进阶_连接与关系_设置数据流负载均衡_设置背压_设置展现弯曲_介绍以及实际操作---大数据之Nifi工作笔记0027
    查看>>
    NIFI数据库同步_多表_特定表同时同步_实际操作_MySqlToMysql_可推广到其他数据库_Postgresql_Hbase_SqlServer等----大数据之Nifi工作笔记0053
    查看>>
    NIFI汉化_替换logo_二次开发_Idea编译NIFI最新源码_详细过程记录_全解析_Maven编译NIFI避坑指南001---大数据之Nifi工作笔记0068
    查看>>
    NIFI汉化_替换logo_二次开发_Idea编译NIFI最新源码_详细过程记录_全解析_Maven编译NIFI避坑指南002---大数据之Nifi工作笔记0069
    查看>>
    NIFI集群_内存溢出_CPU占用100%修复_GC overhead limit exceeded_NIFI: out of memory error ---大数据之Nifi工作笔记0017
    查看>>