nodejs 14.0.0源码分析之setTimeout

2024-03-27 21:32

本文主要是介绍nodejs 14.0.0源码分析之setTimeout,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这一篇我们来看看nodejs是如何实现定时器的。14.0.0的nodejs对定时器模块进行了重构,之前版本的实现是用一个map,以超时时间为键,每个键对应一个队列。即有同样超时时间的节点在同一个队列。每个队列对应一个底层的一个节点(二叉堆里的节点),nodejs在时间循环的timer阶段会从二叉堆里找出超时的节点,然后执行回答,回调里会遍历队列,哪个节点超时了。14.0.0重构后,只使用了一个二叉堆的节点。我们看一下他的实现。
我们先看下定时器模块的组织结构。
在这里插入图片描述
下面我们继续看一下定时器模块的几个重要的数据结果。

1 TimersList

超时时间一样的会被放到同一个队列,这个队列就是由TimersList来管理。对应图中的list那个方框。

// expiry是超时时间的绝对值。用来记录队列中最快到期的节点的时间,msecs是超时时间的相对值(相对插入时的当前时间) 
function TimersList(expiry, msecs) {// 用于链表this._idleNext = this; this._idlePrev = this; this.expiry = expiry;this.id = timerListId++;this.msecs = msecs;// 在优先队列里的位置this.priorityQueuePosition = null;
}

2 优先队列

const timerListQueue = new PriorityQueue(compareTimersLists, setPosition)

nodejs用优先队列对所有1中的链表进行管理,优先队列本质是一个二叉堆(小根堆),每个链表在二叉堆里对应一个节点。根据1中,我们知道每个链表都保存链表中最快到期的节点的过期时间。二叉堆以该事件为依据,即最快到期的list对应二叉堆中的根节点。我们判断根节点是否超时,如果没有超时,说明整个二叉堆的节点都没有超时。如果超时了,就需要不断遍历堆中的节点。

3 超时时间和链表的映射

1中已经提到,超时时间一样的节点,会排在同一个链表中个,nodejs中用一个map保存了超时时间到链表的映射关系。

了解完定时器整体的组织和基础数据结构,我们可以开始进入真正的源码分析了。

我们直接从setTimeout函数开始。

function setTimeout(callback, after, arg1, arg2, arg3) {if (typeof callback !== 'function') {throw new ERR_INVALID_CALLBACK(callback);}let i, args;switch (arguments.length) {// fast casescase 1:case 2:break;case 3:args = [arg1];break;case 4:args = [arg1, arg2];break;default:args = [arg1, arg2, arg3];for (i = 5; i < arguments.length; i++) {// Extend array dynamically, makes .apply run much faster in v6.0.0args[i - 2] = arguments[i];}break;}const timeout = new Timeout(callback, after, args, false, true);insert(timeout, timeout._idleTimeout);return timeout;
}

两个主要操作,new Timeout和insert。我们一个个来。

1 Timeout

function Timeout(callback, after, args, isRepeat, isRefed) {after *= 1; // Coalesce to number or NaNif (!(after >= 1 && after <= TIMEOUT_MAX)) {if (after > TIMEOUT_MAX) {process.emitWarning(`${after} does not fit into` +' a 32-bit signed integer.' +'\nTimeout duration was set to 1.','TimeoutOverflowWarning');}after = 1; // Schedule on next tick, follows browser behavior}// 超时时间相对值this._idleTimeout = after;// 前后指针,用于链表this._idlePrev = this;this._idleNext = this;// 定时器的开始时间this._idleStart = null;// This must be set to null first to avoid function tracking// on the hidden class, revisit in V8 versions after 6.2// 超时回调this._onTimeout = null;this._onTimeout = callback;// 执行回调时传入的参数this._timerArgs = args;// 是否定期执行回调,用于setIntervalthis._repeat = isRepeat ? after : null;this._destroyed = false;// 激活底层的定时器节点(二叉堆的节点),说明有定时节点需要处理if (isRefed)incRefCount();this[kRefed] = isRefed;initAsyncResource(this, 'Timeout');
}

Timeout主要是新建一个对象记录一些定时器的上下文信息。

2 insert(对照上面的图理解)

function insert(item, msecs, start = getLibuvNow()) {msecs = MathTrunc(msecs);// 记录定时器的开始时间,见Timeout函数的定义item._idleStart = start;// 该超时时间是否已经存在对应的链表let list = timerListMap[msecs];// 还没有if (list === undefined) {// 算出绝对超时时间const expiry = start + msecs;// 新建一个链表timerListMap[msecs] = list = new TimersList(expiry, msecs);// 插入优先队列timerListQueue.insert(list);// 算出下一次超时的时间,即最快到期的时间if (nextExpiry > expiry) {// 设置底层的最后超时时间,这样保证可以尽量按时执行scheduleTimer(msecs);nextExpiry = expiry;}}// 把当前节点加到队列里L.append(list, item);
}

scheduleTimer函数是对c++函数的封装。

void ScheduleTimer(const FunctionCallbackInfo<Value>& args) {auto env = Environment::GetCurrent(args);env->ScheduleTimer(args[0]->IntegerValue(env->context()).FromJust());
}void Environment::ScheduleTimer(int64_t duration_ms) {if (started_cleanup_) return;uv_timer_start(timer_handle(), RunTimers, duration_ms, 0);
}

uv_timer_start就是开启底层计时,即往libuv的二叉堆插入一个节点。超时时间是duration_ms,就是最快到期的时间,在timer阶段会判断是否过期。是的话执行RunTimers函数。我们先看一下该函数的主要代码。

Local<Function> cb = env->timers_callback_function();
ret = cb->Call(env->context(), process, 1, &arg);

RunTimers会执行timers_callback_function。timers_callback_function是在nodejs初始化的时候设置的。我们先暂定一下,看一下定时器模块的初始化流程。再回来这里分析。

nodejs在初始化的时候通过一下代码对定时器进行了初始化工作。

setupTimers(processImmediate, processTimers);

setupTimers对应的c++函数是

void SetupTimers(const FunctionCallbackInfo<Value>& args) {auto env = Environment::GetCurrent(args);env->set_immediate_callback_function(args[0].As<Function>());env->set_timers_callback_function(args[1].As<Function>());
}

nodejs把processTimers设置为超时的回调函数。现在我们知道了nodejs是如何设置超时的处理函数,也知道了什么时候会执行该回调。那我们就来看一下回调时具体处理逻辑。

void Environment::RunTimers(uv_timer_t* handle) {Local<Function> cb = env->timers_callback_function();MaybeLocal<Value> ret;Local<Value> arg = env->GetNow();do {// 执行js回调,即下面的processTimers函数ret = cb->Call(env->context(), process, 1, &arg);} while (ret.IsEmpty() && env->can_call_into_js());// 是否执行了所有的节点if (ret.IsEmpty())return;int64_t expiry_ms = ret.ToLocalChecked()->IntegerValue(env->context()).FromJust();uv_handle_t* h = reinterpret_cast<uv_handle_t*>(handle);// 还有超时节点,开块超时时间是expiry_ms ,需要重新插入底层的二叉堆。if (expiry_ms != 0) {// 算出下次超时的相对值int64_t duration_ms =llabs(expiry_ms) - (uv_now(env->event_loop()) - env->timer_base());// 重新把handle插入libuv的二叉堆env->ScheduleTimer(duration_ms > 0 ? duration_ms : 1);}
}

该函数主要是执行回调,然后如果还有没超时的节点,重新设置libuv定时器的时间。看看js层面。

  function processTimers(now) {nextExpiry = Infinity;let list;let ranAtLeastOneList = false;// 取出优先队列的根节点,即最快到期的节点while (list = timerListQueue.peek()) {// 还没过期,if (list.expiry > now) {nextExpiry = list.expiry;// 返回下一次过期的时间return refCount > 0 ? nextExpiry : -nextExpiry;}listOnTimeout(list, now);}return 0;}function listOnTimeout(list, now) {const msecs = list.msecs;debug('timeout callback %d', msecs);let ranAtLeastOneTimer = false;let timer;// 遍历具有统一相对过期时间的队列while (timer = L.peek(list)) {// 算出已经过去的时间const diff = now - timer._idleStart;// 过期的时间比超时时间小,还没过期if (diff < msecs) {// 整个链表节点的最快过期时间等于当前还没过期节点的值,链表是有序的list.expiry = MathMax(timer._idleStart + msecs, now + 1);// 更新id,用于决定在优先队列里的位置list.id = timerListId++;// 调整过期时间后,当前链表对应的节点不一定是优先队列里的根节点了,可能有他更快到期,即当前链表需要往下沉timerListQueue.percolateDown(1);return;}// 准备执行用户设置的回调,删除这个节点L.remove(timer);let start;if (timer._repeat)start = getLibuvNow();try {const args = timer._timerArgs;// 执行用户设置的回调if (args === undefined)timer._onTimeout();elsetimer._onTimeout(...args);} finally {// 设置了重复执行回调,即来自setInterval。则需要重新加入链表。if (timer._repeat && timer._idleTimeout !== -1) {// 更新超时时间,一样的时间间隔timer._idleTimeout = timer._repeat;// 重新插入链表insert(timer, timer._idleTimeout, start);} else if (!timer._idleNext && !timer._idlePrev && !timer._destroyed) {timer._destroyed = true;if (timer[kRefed])refCount--;}// 为空则删除if (list === timerListMap[msecs]) {delete timerListMap[msecs];timerListQueue.shift();}}

上面的代码主要是遍历优先队列,

  • 如果当前节点超时,即遍历他对应的链表。否则重新计算出最快超时时间,修改底层libuv的节点。即更新超时时间。
  • 遍历链表的时候如果遇到超时的则执行,如果没有超时的说明后面的节点也不会超时了。因为链表是有序的。修改链表的最快超时时间的值,调整他在优先队列的位置。因为超时时间变了。可能需要调整。

定时器模块的setTimeout分析完了,后面有机会的话再补充一下,另外setInterval是类似的。

这篇关于nodejs 14.0.0源码分析之setTimeout的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/853452

相关文章

[职场] 公务员的利弊分析 #知识分享#经验分享#其他

公务员的利弊分析     公务员作为一种稳定的职业选择,一直备受人们的关注。然而,就像任何其他职业一样,公务员职位也有其利与弊。本文将对公务员的利弊进行分析,帮助读者更好地了解这一职业的特点。 利: 1. 稳定的职业:公务员职位通常具有较高的稳定性,一旦进入公务员队伍,往往可以享受到稳定的工作环境和薪资待遇。这对于那些追求稳定的人来说,是一个很大的优势。 2. 薪资福利优厚:公务员的薪资和

springboot家政服务管理平台 LW +PPT+源码+讲解

3系统的可行性研究及需求分析 3.1可行性研究 3.1.1技术可行性分析 经过大学四年的学习,已经掌握了JAVA、Mysql数据库等方面的编程技巧和方法,对于这些技术该有的软硬件配置也是齐全的,能够满足开发的需要。 本家政服务管理平台采用的是Mysql作为数据库,可以绝对地保证用户数据的安全;可以与Mysql数据库进行无缝连接。 所以,家政服务管理平台在技术上是可以实施的。 3.1

高仿精仿愤怒的小鸟android版游戏源码

这是一款很完美的高仿精仿愤怒的小鸟android版游戏源码,大家可以研究一下吧、 为了报复偷走鸟蛋的肥猪们,鸟儿以自己的身体为武器,仿佛炮弹一样去攻击肥猪们的堡垒。游戏是十分卡通的2D画面,看着愤怒的红色小鸟,奋不顾身的往绿色的肥猪的堡垒砸去,那种奇妙的感觉还真是令人感到很欢乐。而游戏的配乐同样充满了欢乐的感觉,轻松的节奏,欢快的风格。 源码下载

高度内卷下,企业如何通过VOC(客户之声)做好竞争分析?

VOC,即客户之声,是一种通过收集和分析客户反馈、需求和期望,来洞察市场趋势和竞争对手动态的方法。在高度内卷的市场环境下,VOC不仅能够帮助企业了解客户的真实需求,还能为企业提供宝贵的竞争情报,助力企业在竞争中占据有利地位。 那么,企业该如何通过VOC(客户之声)做好竞争分析呢?深圳天行健企业管理咨询公司解析如下: 首先,要建立完善的VOC收集机制。这包括通过线上渠道(如社交媒体、官网留言

基于Java医院药品交易系统详细设计和实现(源码+LW+调试文档+讲解等)

💗博主介绍:✌全网粉丝10W+,CSDN作者、博客专家、全栈领域优质创作者,博客之星、平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌💗 🌟文末获取源码+数据库🌟 感兴趣的可以先收藏起来,还有大家在毕设选题,项目以及论文编写等相关问题都可以给我留言咨询,希望帮助更多的人  Java精品实战案例《600套》 2023-2025年最值得选择的Java毕业设计选题大全:1000个热

美容美发店营销版微信小程序源码

打造线上生意新篇章 一、引言:微信小程序,开启美容美发行业新纪元 在数字化时代,微信小程序以其便捷、高效的特点,成为了美容美发行业营销的新宠。本文将带您深入了解美容美发营销微信小程序,探讨其独特优势及如何助力商家实现业务增长。 二、微信小程序:美容美发行业的得力助手 拓宽客源渠道:微信小程序基于微信社交平台,轻松实现线上线下融合,帮助商家快速吸引潜在客户,拓宽客源渠道。 提升用户体验:

风水研究会官网源码系统-可展示自己的领域内容-商品售卖等

一款用于展示风水行业,周易测算行业,玄学行业的系统,并支持售卖自己的商品。 整洁大气,非常漂亮,前端内容均可通过后台修改。 大致功能: 支持前端内容通过后端自定义支持开启关闭会员功能,会员等级设置支持对接官方支付支持添加商品类支持添加虚拟下载类支持自定义其他类型字段支持生成虚拟激活卡支持采集其他站点文章支持对接收益广告支持文章评论支持积分功能支持推广功能更多功能,搭建完成自行体验吧! 原文

HTML5文旅文化旅游网站模板源码

文章目录 1.设计来源文旅宣传1.1 登录界面演示1.2 注册界面演示1.3 首页界面演示1.4 文旅之行界面演示1.5 文旅之行文章内容界面演示1.6 关于我们界面演示1.7 文旅博客界面演示1.8 文旅博客文章内容界面演示1.9 联系我们界面演示 2.效果和源码2.1 动态效果2.2 源代码2.3 源码目录 源码下载万套模板,程序开发,在线开发,在线沟通 作者:xcLeigh

打包体积分析和优化

webpack分析工具:webpack-bundle-analyzer 1. 通过<script src="./vue.js"></script>方式引入vue、vuex、vue-router等包(CDN) // webpack.config.jsif(process.env.NODE_ENV==='production') {module.exports = {devtool: 'none

Java中的大数据处理与分析架构

Java中的大数据处理与分析架构 大家好,我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编,也是冬天不穿秋裤,天冷也要风度的程序猿!今天我们来讨论Java中的大数据处理与分析架构。随着大数据时代的到来,海量数据的存储、处理和分析变得至关重要。Java作为一门广泛使用的编程语言,在大数据领域有着广泛的应用。本文将介绍Java在大数据处理和分析中的关键技术和架构设计。 大数据处理与