前端知识体系-NodeJS相关】NodeJS基础知识全面总结
nodejs基础知识
1. node的全局对象和全局变量
1.1 全局对象:所有模块都可以调用的
global:表示node所在的全局环境,类似于浏览器的window对象。
process:该对象表示node所处的当前进程,允许开发者与该进程互动。
console:指向node内置的console模块,提供命令行环境中的标准输入、标准输出功能。
1.2 全局函数
- 定时器函数:共有4个,分别是settimeout(), cleartimeout(), setinterval(), clearinterval();
- require:用于加载模块;
- buffer():用于操作二进制数据。
1.3 全局变量
- __filename:指向当前运行的脚本文件名。
- __dirname:指向当前运行的脚本所在的目录。
2. node的三大特点
2.1 单线程
node.js不为每个客户连接创建一个新的线程,而仅仅使用一个线程。当有用户连接了,就触发一个内部事件,通过非阻塞i/o、事件驱动机制,让node.js程序宏观上也是并行的。
2.2 非阻塞i/o
由于node.js中采用了非阻塞型i/o机制,因此在执行了访问数据库的代码之后,将立即转而执行其后面的代码,把数据库返回结果的处理代码放在回调函数中,从而提高了程序的执行效率。
当某个i/o执行完毕时,将以事件的形式通知执行i/o操作的线程,线程执行这个事件的回调函数。为了处理异步i/o,线程必须有事件循环,不断的检查有没有未处理的事件,依次予以处理。
阻塞模式下,一个线程只能处理一项任务,要想提高吞吐量必须通过多线程。而非阻塞模式下,一个线程永远在执行计算操作,这个线程的cpu核心利用率永远是100%。所以,这是一种特别有哲理的解决方案:与其人多,但是好多人闲着;还不如一个人玩命,往死里干活儿。
2.3 事件驱动event-driven
在node中,在一个时刻,只能执行一个事件回调函数,但是在执行一个事件回调函数的中途,可以转而处理其他事件(比如,又有新用户连接了),然后返回继续执行原事件的回调函数,这种处理机制,称为“事件环”机制。
node.js底层是c++(v8也是c++写的)。底层代码中,近半数都用于事件队列、回调函数队列的构建。
3. node技术架构
3.1 node底层架构
nodejs组成部分:v8 engine, libuv, builtin modules, native modules以及其他辅助服务。
v8 engine:主要有两个作用
1.虚拟机的功能,执行js代码(自己的代码,第三方的代码和native modules的代码)。
2.提供c++函数接口,为nodejs提供v8初始化,创建context,scope等。libuv:它是基于事件驱动的异步io模型库,我们的js代码发出请求,最终由libuv完成,而我们所设置的回调函数则是在libuv触发。
builtin modules:它是由c++代码写成各类模块,包含了crypto,zlib, file stream etc 基础功能。(v8提供了函数接口,libuv提供异步io模型库,以及一些nodejs函数,为builtin modules提供服务)。
native modules:它是由js写成,提供我们应用程序调用的库,同时这些模块又依赖builtin modules来获取相应的服务支持
[!note]
总结:如果把nodejs看做一个黑匣子,起暴露给开发者的接口则是native modules,当我们发起请求时,请求自上而下,穿越native modules,通过builtin modules将请求传送至v8,libuv和其他辅助服务,请求结束,则从下回溯至上,最终调用我们的回调函数。
3.2 node函数调用机制
v8执行js代码 server.listen()时,会通过一些基础服务到tcpwrap::listen(),tcpwrap是nodejs的內建模块,其通过libuv的api uv_listen()的方式,由libuv来完成异步调用。
图中1,2,3,4,5步骤标明了调用和返回的路径,这几步很快结束,留下callback tcpwrap::onconnection()等着所需要的数据准备好后被调用。
libuv在得到所需要的请求后,会调用callback tcpwrap::onconnection(),在该函数最后通过 tcp_wrap->makecallback(env->onconnection_string(), array_size(argv), argv) 调用v8 engine中的javascript callback。
node.js内建模块http其实是建立在模块net之上的。如果看net.js代码会发现,其通过 new tcp() 返回的类对象完成后续的tcp connect, bind, open等socket动作。
可以看到node.js做的工作像是一座桥。左手v8,右手libuv,将2者有机连接在一起。例如handlewrap::handlewrap()中记录了v8 instance中的javascript对象以及tcpwrap对象。这样在tcpwrap::onconnection()中可以拿到这两个对象,执行后续的callback调用。
参考文章