当前位置: 首页 > news >正文

广西做网站/搜索引擎官网

广西做网站,搜索引擎官网,哈尔滨网站备案手续费,北京日报客户端疫情上一篇分析了prepare阶段,check和idle阶段是一样的,所以就不分析了。今天分析定时器阶段。nodejs中setTimeout和setInterval就是使用libuv的定时器阶段实现的。libuv中,定时器是以最小堆实现的。即最快过期的节点是根节点。我看看看定时器的数…

上一篇分析了prepare阶段,check和idle阶段是一样的,所以就不分析了。今天分析定时器阶段。nodejs中setTimeout和setInterval就是使用libuv的定时器阶段实现的。libuv中,定时器是以最小堆实现的。即最快过期的节点是根节点。我看看看定时器的数据结构。

8361f3373c9eff3ba04b3babb474be98.png

看一下定时器的使用。

int main() v_timer_t once;uv_timer_init(uv_default_loop(), &once);uv_timer_start(&once, once_cb, 10, 0);uv_run(uv_default_loop(), UV_RUN_DEFAULT);return 0;
}

我们从uv_timer_init函数开始分析。

// 初始化uv_timer_t结构体
int uv_timer_init(uv_loop_t* loop, uv_timer_t* handle) {uv__handle_init(loop, (uv_handle_t*)handle, UV_TIMER);handle->timer_cb = NULL;handle->repeat = 0;return 0;
}

init函数和其他阶段的init函数一样,初始化handle和私有的一些字段。接着我们看start函数。该函数是启动一个定时器(省略部分代码)。

// 启动一个计时器
int uv_timer_start(uv_timer_t* handle,uv_timer_cb cb,uint64_t timeout,uint64_t repeat) {uint64_t clamped_timeout;// 重新执行start的时候先把之前的停掉if (uv__is_active(handle))uv_timer_stop(handle);// 超时时间,为绝对值clamped_timeout = handle->loop->time + timeout;// 初始化回调,超时时间,是否重复计时,赋予一个独立无二的idhandle->timer_cb = cb;handle->timeout = clamped_timeout;handle->repeat = repeat;/* start_id is the second index to be compared in uv__timer_cmp() */handle->start_id = handle->loop->timer_counter++;// 插入最小堆heap_insert(timer_heap(handle->loop),(struct heap_node*) &handle->heap_node,timer_less_than);// 激活该handleuv__handle_start(handle);return 0;
}

start函数首先初始化handle里的某些字段,包括超时回调,是否重复启动定时器、超时的绝对时间等。接着把handle节点插入到最小堆中。最后给这个handle打上标记,激活这个handle。这时候的结构体如下。

1e8b763aff6b081cb8df8bdcca94b89b.png

这时候到了事件循环的timer阶段。

// 找出已经超时的节点,并且执行里面的回调
void uv__run_timers(uv_loop_t* loop) {struct heap_node* heap_node;uv_timer_t* handle;for (;;) {heap_node = heap_min(timer_heap(loop));if (heap_node == NULL)break;handle = container_of(heap_node, uv_timer_t, heap_node);// 如果当前节点的时间大于当前时间则返回,说明后面的节点也没有超时if (handle->timeout > loop->time)break;// 移除该计时器节点,重新插入最小堆,如果设置了repeat的话uv_timer_stop(handle);uv_timer_again(handle);// 执行超时回调handle->timer_cb(handle);}
}

libuv在每次事件循环开始的时候都会缓存当前的时间,在整个一轮的事件循环中,使用的都是这个缓存的时间。缓存了当前最新的时间后,就执行uv__run_timers,该函数的逻辑很明了,就是遍历最小堆,找出当前超时的节点。因为堆的性质是父节点肯定比孩子小。所以如果找到一个节点,他没有超时,则后面的节点也不会超时。对于超时的节点就知道他的回调。执行完回调后,还有两个关键的操作。第一就是stop,第二就是again。

// 停止一个计时器
int uv_timer_stop(uv_timer_t* handle) {if (!uv__is_active(handle))return 0;// 从最小堆中移除该计时器节点heap_remove(timer_heap(handle->loop),(struct heap_node*) &handle->heap_node,timer_less_than);// 清除激活状态和handle的active数减一uv__handle_stop(handle);return 0;
}

stop的逻辑很简单,其实就是把handle从二叉堆中删除。并且取消激活状态。那么againt又是什么呢?again是为了支持setInterval这种场景。

// 重新启动一个计时器,需要设置repeat标记 
int uv_timer_again(uv_timer_t* handle) {// 如果设置了repeat标记说明计时器是需要重复触发的if (handle->repeat) {// 先把旧的计时器节点从最小堆中移除,然后再重新开启一个计时器uv_timer_stop(handle);uv_timer_start(handle, handle->timer_cb, handle->repeat, handle->repeat);}return 0;
}

如果handle设置了repeat标记,则该handle在超时后,每repeat的时间后,就会继续执行超时回调。对于setInterval,就是超时时间是x,每x的时间后,执行回调。这就是nodejs里定时器的底层原理。但nodejs不是每次调setTimeout的时候都往最小堆插入一个节点。nodejs里,只有一个关于uv_timer_s的handle。他在js层维护了一个数据结构,每次计算出最早到期的节点,然后修改handle的超时时间。具体原理在之前的一篇文章已经分析过。
timer阶段和poll io阶段也有一些联系,因为poll io可能会导致主线程阻塞,为了保证主线程可以尽快执行定时器的回调,poll io不能一直阻塞,所以这时候,阻塞的时长就是最快到期的定时器节点的时长。

http://www.lbrq.cn/news/1277749.html

相关文章:

  • 网站制作完成/如何创建网站站点
  • 网站建设费汇算清缴/电脑优化大师下载安装
  • 郑州网站推广平台/网络舆情优化公司
  • 上海有多少家公司/百度的seo关键词优化怎么弄
  • 政府网站建设ppt/sem优化软件哪家好
  • 成都住建局官网首页/海外seo网站推广
  • 珠海网站建设维护/如何网络推广新产品
  • 赌博游戏网站怎么做/制作自己的网页
  • 中山市住房建设局网站/seo属于运营还是技术
  • 被关闭的设计网站/搜索引擎推广方案
  • 品牌建设的三大理论/外贸网站推广seo
  • 在线制作logo图片/网站seo哪家好
  • 想学做网站学什么教程/网络推广哪个平台最好
  • 黄冈网站推广都有哪些渠道/百度一下官方网站
  • 上海 企业网站建设/短视频推广引流方案
  • 网络营销导向企业网站建设的原则包括/西安seo外包行者seo06
  • 网站流量一直下降/去了外包简历就毁了吗
  • 网站建设谈判技巧/推广网站都有哪些
  • 淮安市城市建设档案馆网站/完整的品牌推广方案
  • 厦门今天刚刚发生的新闻/seo推广怎么做视频教程
  • 花生壳做网站/免费舆情监测平台
  • 做网页设计网站有哪些/邯郸seo优化
  • 手机触屏网站开发/西安seo优化培训机构
  • 网站首页被k 不恢复/上往建站
  • 重新安wordpress网站/seo优化教程下载
  • 网站建设中的需求报告功能/推广app接单网
  • bootstrap建设淘宝网站/自己怎么做一个网页
  • 百度联盟做网站赚钱吗/图片百度搜索
  • 银川森林半岛/深圳seo优化服务商
  • 网站维护什么情况/周口搜索引擎优化
  • 《React与Vue构建TODO应用的深层逻辑》
  • 05 OpenCV--图像预处理之图像轮廓、直方图均衡化、模板匹配、霍夫变化、图像亮度变化、形态学变化
  • 第四章:分析 Redis 性能高原因和核心字符串类型命令
  • 昇思学习营-模型推理和性能优化
  • 论文阅读-IGEV
  • 通信名词解释:I2C、USART、SPI、RS232、RS485、CAN、TCP/IP、SOCKET、modbus