Node.js 源码解读 EventLoop
之前断断续续开发过一些 Node.js 的项目,但只仅限于使用它实现一些功能,没有过多对底层深入的研究。现在因为公司大前端组内的服务端渲染直出、BFF(Backend For Frontend) 等需求会越来越多,组内需要对服务端技术有更深刻的理解,如果对 Node.js 仅仅停留在如何写业务代码的层面,那恐怕是没有底气保证以后服务的稳定性。
本文会基于 node-v12.13.0 版本的源码,对核心模块代码做一些阅读和理解,以窥探 Node.js 服务高效的秘诀。在研究源码之前,首先带着几个疑问,看接下来是否能一一解开:
- EventLoop 的任务调度方式是什么样的?一次 Loop 取一个任务还是多个任务?
- 主循环在没有任务处理的空闲时,如何休眠的?是像 iOS 的 runloop 调用系统进程挂起吗?
事件循环的实现是属于 libuv 核心库的一部分,而在 node 项目中,他的位置是在 deps/uv/src/ 目录下,打开 core.c 文件,其中的 uv_run() 就是主循环的入口函数,整个函数的实现也基本和官方文档给出的事件循环阶段一致。

源码

当要写这篇博客的时候,实际上已经有很多介绍这些内容的文章了,所以本篇就直接略过一些 loop 阶段,直接讲最重要的:
1.计算超时时间
超时时间直接决定了 poll 阶段是阻塞还是非阻塞的直接执行,所以他决定着任务调度的实际执行时机和执行方式,主要实现在 uv_backend_timeout() 函数里
展开函数的实现:
int uv_backend_timeout(const uv_loop_t* loop) {
  // loop 将要停止时,返回0
  if (loop->stop_flag != 0)
    return 0;
 // 没有活跃的 handles 时,返回0
  if (!uv__has_active_handles(loop) && !uv__has_active_reqs(loop))
    return 0;
 // idle不为空时,返回0
  if (!QUEUE_EMPTY(&loop->idle_handles))
    return 0;
 // pending_queue(执行任务已完成待回调队列)不为空时,返回0
  if (!QUEUE_EMPTY(&loop->pending_queue))
    return 0;
 // closeing_handles 不为空时,返回0
  if (loop->closing_handles)
    return 0;
  return uv__next_timeout(loop);
}
接下来是 uv_next_timeout() 这个函数,它是实现在 deps/uv/src/time.c 文件里:
int uv__next_timeout(const uv_loop_t* loop) {
  const struct heap_node* heap_node;
  const uv_timer_t* handle;
  uint64_t diff;
  heap_node = heap_min(timer_heap(loop));
 // timer 队列为空时,返回-1(-1的含义在下面介绍)
  if (heap_node == NULL)
    return -1; /* block indefinitely */
  handle = container_of(heap_node, uv_timer_t, heap_node);
 // 如果 timer 超时了,返回0
  if (handle->timeout <= loop->time)
    return 0;
 // 将超时时间设为最早要超时的 timer 的所剩余时间
  diff = handle->timeout - loop->time;
  if (diff > INT_MAX)
    diff = INT_MAX;
  return (int) diff;
}
从上面的函数计算好的 timeout 将以参数的形式传入 uv__io_poll() 这个函数,这个函数内就是 poll 阶段的实现了,我已经迫不及待的要一探究竟了。uv__io_poll() 依赖的操作系统提供的功能,具有平台相关性,所以不同的平台会有不同的实现,本文主要讨论linux-core.h这个文件,即linux平台的实现。
代码比较长我就不贴了,总之看完并理解下来,uv_io_poll 就是对 epoll 的一个封装,但uv给我们提供了一个很值得思考和借鉴的方法,那就是 timeout 的使用,根据这个 timeout 来动态决定一次 loop 将要处理的任务量。
首选来解析一下 epoll_pwait 这个函数的作用
.....
......
nfds = epoll_pwait(loop->backend_fd,
events,
ARRAY_SIZE(events),
timeout,
psigset);
......
......
所以从任务队列取并不是简单的只取一条,在这个超时时间内,可能会在一次 loop 的 poll 阶段完成多个任务,完成后会立即回调【阻塞式的完成】。
(未完待续)
p.p1 { margin: 0; font: 12px "Helvetica Neue" }
li.li1 { margin: 0; font: 12px ".PingFang SC" }
span.s1 { font: 12px "Helvetica Neue" }
ol.ol1 { list-style-type: decimal }
Node.js 源码解读 EventLoop的更多相关文章
- js便签笔记(10) - 分享:json2.js源码解读笔记
		1. 如何理解“json” 首先应该意识到,json是一种数据转换格式,既然是个“格式”,就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转 ... 
- js便签笔记(10) - 分享:json.js源码解读笔记
		1. 如何理解“json” 首先应该意识到,json是一种数据转换格式,既然是个“格式”,就是个抽象的东西.它不是js对象,也不是字符串,它只是一种格式,一种规定而已. 这个格式规定了如何将js对象转 ... 
- fastclick.js源码解读分析
		阅读优秀的js插件和库源码,可以加深我们对web开发的理解和提高js能力,本人能力有限,只能粗略读懂一些小型插件,这里带来对fastclick源码的解读,望各位大神不吝指教~! fastclick诞生 ... 
- Linux 安装node.js ---- 源码编译的方式
		一 : 普通用户: 安装前准备环境: 1.检查Linux 版本 命令: cat /etc/redhat-release 2.检查 gcc.gcc-c++ 是否安装过 命令: rpm -q gcc rp ... 
- prototype.js 源码解读(02)
		如果你想研究一些比较大型的js框架的源码的话,本人建议你从其最初的版本开始研读,因为最初的版本东西少,易于研究,而后的版本基本都是在其基础上不断扩充罢了,所以,接下来我不准备完全解读prototype ... 
- prototype.js 源码解读(01)
		prototype.js是一个设计的非常优雅且很有实用价值的js基础类库,其源码非常值得研究.研究它的源码不仅能提升个人水平,而且对你打下坚实的js基础也很有帮助.因本人技术水平有限,该解读仅供参考. ... 
- json2.js源码解读记录
		相关内容:json详细用法.js语法.unicode.正则 json特点--最简单.最小巧的经典js库. json作者:道克拉斯.克劳福德(Douglas Crockford)--js大牛 出 ... 
- require.js 源码解读——配置默认上下文
		首先,我们先来简单说一下,require.js的原理: 1.载入模块 2.通过模块名解析出模块信息,以及计算出URL 3.通过创建SCRIPT的形式把模块加载到页面中. 4.判断被加载的脚本,如 ... 
- Red hat Linux 安装Node.js 源码安装
		1. 下载源码包 http://nodejs.org/dist/v0.10.29/node-v0.10.29.tar.gz 2.准备安装环境,>python2.6, gcc, g++ pytho ... 
- 亚马逊左侧菜单延迟z三角 jquery插件jquery.menu-aim.js源码解读
		关于亚马逊的左侧菜单延迟,之前一直不知道它的实现原理.梦神提到了z三角,我也不知道这是什么东西.13号那天很有空,等领导们签字完我就可以走了.下午的时候,找到了一篇博客:http://jayuh.co ... 
随机推荐
- 修改ctags让fzf.vim插件显示C,C++方法声明的标签
			背景 在 vim 中使用 fzf.vim 插件可以进行方便的搜索文件, 源码TAG, GIT 记录等, 最近抽空看了下 BTags 命令在 c, c++ 文件中, 无法显示头文件中的函数声明 标签问题 ... 
- honoka和格点三角形
			题目: honoka最近在研究三角形计数问题.她认为,满足以下三个条件的三角形是"好三角形".1.三角形的三个顶点均为格点,即横坐标和纵坐标均为整数.2.三角形的面积为 .3.三角 ... 
- ChatGPT API使用介绍
			1.概述 随着人工智能技术的不断发展,越来越多的AI产品被应用到各个领域,其中最具代表性的莫过于人工智能语言模型.语言模型是一种可以通过学习大量语言数据来预测文本或语音的技术,其应用范围十分广泛,如智 ... 
- MyBatis 查询的条目与预期的不一致
			预期查询的数据条目是 4 条: 但是 MyBatis 查询出来的结果只有 2 条数据: resultMap 开启了 autoMapping 功能,就不需要多余地添加 result.下面是错误的映射操作 ... 
- sqlite3数据库Linux 系统移植和使用
			sqlite3数据库是一个小型的数据库,当数据量不大,要求不是特别高的时候,是个不错的选择. 在Linux上移植和使用也非常的方便. 本示例是在硬件全志r528 .linux5.4 上验证的. 移植操 ... 
- pip 基本问题
			pip语法错误 pip换源 更新错误 无模板 Ignoring警告 问题一 (pip语法错误 ) pip list File "<stdin>", line 1 pip ... 
- 【Redis的三种数据删除策略】定时定期惰性,超出内存就自动清理
			https://blog.csdn.net/DQWERww/article/details/126453008 https://blog.csdn.net/qq_38056518/article/de ... 
- flask 项目构建
			创建虚拟环境 mkvirtualenv 环境名 安装所需要的包 pip install flask == 0.12.4 pip install redis pip install flask-ses ... 
- springmvc 能进入Controller,但是前端页面还是404
			问题现象: 1.Controller里面的方法上已经加了注解 @ResponseBody 2.能进入controller的方法里面: 3.前端返回状态码 404: 4.控制台没有异常信息: 问题原因: ... 
- VUE学习-过渡 & 动画
			过渡 & 动画 Vue 在插入.更新或者移除 DOM 时,提供多种不同方式的应用过渡效果.包括以下工具: 在 CSS 过渡和动画中自动应用 class 在过渡钩子函数中使用 JavaScrip ... 
