You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
// lib/timers.js setImmediate回调追踪// ...constImmediate=classImmediate{constructor(callback,args){this._idleNext=null;this._idlePrev=null;// this must be set to null first to avoid function tracking// on the hidden class, revisit in V8 versions after 6.2this._onImmediate=null;this._onImmediate=callback;// 注意这里this._argv=args;this._destroyed=false;this[kRefed]=false;initAsyncResource(this,'Immediate');this.ref();immediateInfo[kCount]++;immediateQueue.append(this);}// ...}// ...
intuv_backend_timeout(constuv_loop_t*loop) {
if (loop->stop_flag!=0)
return0;
if (!uv__has_active_handles(loop) && !uv__has_active_reqs(loop))
return0;
if (!QUEUE_EMPTY(&loop->idle_handles))
return0;
if (!QUEUE_EMPTY(&loop->pending_queue))
return0;
if (loop->closing_handles)
return0;
returnuv__next_timeout(loop);
}
voidEnvironment::ToggleImmediateRef(bool ref) {
if (ref) {
// Idle handle is needed only to stop the event loop from blocking in poll.uv_idle_start(immediate_idle_handle(), [](uv_idle_t*){ });
} else {
uv_idle_stop(immediate_idle_handle());
}
}
前言
相信稍微对node感兴趣的同学都知道setImmediate触发是在event-loop的check阶段,那么这个setImmediate到底是在哪里实现的注册到uv_check中以及如何触发其中的回调呢?
js入口
如果要想撕开setImmediate的口子,那么最简单粗暴的方式就是直接从./lib/timers这里入手。闲话少说,直接撸代码:
这是setImmediate所调用的构造函数的一部分内容,从这里可以很明确的看出来我们注册的回调传递给了
this._onImmediate
。接下来我们看一下最终触发回调的代码:
找起来也是轻松加愉快,通过
_onImmediate
我们很快便找到了触发函数(或者直接找apply
)。这样我们可以通过函数名runCallback
一层一层向上溯源了。最终溯源的结果在这里:
向上溯源的函数是通过
setupTimers
进行注册的,而setupTimers
定睛一看:process.binding
,内置模块timer_wrap
浮出了水面(在这里简单提一下,在最近的pr: timers: refactor timer list processing 中setTimeout
的触发函数也修改为了使用setupTimers
注册)。node中对setImmediate的处理
processImmediate函数在node中的注册
接上文,我们视线转移到./src/timer_wrap.cc的
SetupTimers
函数中:详细介绍一下
env->set_immediate_callback_function(args[0].As<Function>())
。这句话的来源其实在env.h
中:整体意思是:定义了一个带参宏
V
,而这个宏在调用的时候会定义一个属性,所以在通过V(immediate_callback_function, v8::Function)
调用后,可以实现env->set_immediate_callback_function
的调用了,同时还会生成一个成员函数PropertyName()
,所以在调用后同时也会使得env->immediate_callback_function
成为可调用函数。args[0]
在这里指的便是processImmediate
,所以通过SetupTimers
可以使processImmediate
这个函数最终注册到node中。libuv中对setImmediate的处理
processImmediate函数在libuv中的注册
视线转移到src/env.cc中,不知道大家是否还记得在第一章一个简单的nodejs文件从运行到结束都发生了什么中曾经一笔带过
Environment::Start
。没错,setImmediate就是在Environment::Start
中注册到libuv的,接下来看代码:有关于
immediate
的总共有四句,接下来我们逐个详细介绍一下:uv_check_init(event_loop(), immediate_check_handle());
这一句主要是初始化uv_check的handle;uv_unref(reinterpret_cast<uv_handle_t*>(immediate_check_handle()));
这一句主要是解除uv_check的handle在event-loop中的引用,因为我们希望在event-loop中没有活跃handle的时候自动退出;uv_idle_init(event_loop(), immediate_idle_handle());
这里涉及到uv_idle的概念,uv_idle总是在uv_prepare阶段之前运行,在这里是用uv_idle_init
方法对uv_idle的handle进行初始化;uv_check_start(immediate_check_handle(), CheckImmediate);
在这里,真正的把上文提到的processImmediate
通过函数CheckImmediate
注册到了uv_check中。uv_idle简介
在这里涉及到了uv_idle系列api,那么就给大家稍作介绍:
相信很多人都看过node官方那张经典的event-loop的图,在
idle、prepare
阶段的下一阶段是poll
。而poll
阶段是不断轮训执行callback,所以是会阻塞的。具体的调用代码是uv__io_poll(loop, timeout);
,这里的timeout就是超时时间,具体设置timeout的代码可以看这里:可以看到在一些情况下超时时间可以是0——即可以直接跨过poll阶段到达下一个check阶段,而check阶段就是setImmediate执行的阶段。这些可以跨过poll阶段的情况有:
stop_flag
直接强制跨过;uv__io_init
会初始化pending_queue);setImmediate正是利用了idle,实现了对poll阶段的跨越。
setImmediate与uv_idle
在src/env.cc中有一个比较不起眼的api--
ToggleImmediateRef
:不知道大家还记得上文提到过的
SetupTimers
吗,里面有一行代码:结合这两个函数,很容易得出结论:
setImmediate
通过函数ToggleImmediateRef
对uv_idle进行开关的控制,开的时候可以直接越过poll阶段,关的时候则执行poll阶段。setImmediate的执行
刚才聊到了,通过
uv_check_start(immediate_check_handle(), CheckImmediate);
把setImmediate
的上层函数processImmediate
通过CheckImmediate
注册到了uv_check中。接下来我们看下CheckImmediate
:相信如果读者从头到尾贯穿下来的话,这里已经很明了了,通过一个
do...while...
实现了对immediate_callback_function
的调用,即调用了js中的processImmediate
进而实现了setImmediate
的运行。在运行完后,通过env->ToggleImmediateRef(false);
实现对uv_idle
的停止,进而使得poll能阻塞处理回调。结语
通过上面的分析,读者基本可以清晰了解到
setImmediate
的整体注册和触发流程。而真正触发evnet-loop的,则是在src/node.cc中:by 小菜
The text was updated successfully, but these errors were encountered: