node核心特性理解

原文地址在我的博客,转载请注明来源,谢谢!

node是在前端领域经常看到的词。node对于前端的重要性已经不言而喻,掌握node也是作为合格的前端工程师一项基本功了。知道node、知道后端的一些东西,才能更好的与别人合作,发挥更大的价值。

概述

本文主要介绍了我对node的一些核心特性的理解,包括node架构、特点、机制、核心模块与简单应用。

正文

从浏览器到node

首先,node是一个平台,使用javascript作为编程语言,运行在服务端。服务端语言能做的,node一般都能做,而且有些情况下做的更好,因为它具有自己的特色。

node是javascript运行环境(runtime),就像浏览器一样,是一个平台。在浏览器中,V8引擎负责解释javascript,你在javascript调用的接口都是浏览器实现并提供的,浏览器会调用底层的、由其他语言(C++)实现并封装好的接口来完成任务;同样,在node中,也是V8引擎负责解释javascript,而你在javascript调用的浏览器提供的接口就不能用了,因为它脱离了浏览器的环境,但是因为你在node环境中,你就可以使用node提供的由C++语言实现的、由javascript封装好的各种接口来完成后端任务。浏览器提供的API用于处理前端任务,比如弹个窗,换个主题,处理用户操作等,而node因为服务后端,因此提供的API则用来处理后端任务,比如响应请求,读取文件等,这些API由不同的模块提供。因为关注领域不一样,因此所做的任务就不一样,提供的API就不一样,但是原理、相关实现大致与浏览器端相同。

从浏览器到node这一块如果想了解更多,推荐IBM的文章node.js到底是什么?

node 架构

node架构分为三层(参考链接):

图片来源

  • Node standard library:node标准库,也就是node模块提供各种接口的javascript实现,任何javascript代码、npm install 或者你写的模块都在这里
  • Node bindings:包括C/C++ bindings(胶水代码)和Add on(添加其他C/C++库时需要自己写的Bindings),这一层向下封装了V8和libuv接口,向上提供了基础API接口,是连接javascript和C++的桥梁
  • 第三层是支撑 Node.js 运行的关键,由 C/C++ 实现。
    • V8 是Google开发的JavaScript引擎,提供JavaScript运行环境,可以说它就是 Node.js 的发动机,负责解释javascript,与chrome浏览器相同。
    • Libuv 是专门为Node.js开发的一个封装库,提供跨平台的异步I/O能力,负责node运行时的线程池调度。
    • C-ares:提供了异步处理 DNS 相关的能力。
    • http_parser、OpenSSL、zlib 等:提供包括 http 解析、SSL、数据压缩等系统底层的访问。

平常我们用到的也就是第一层node各个模块实现的接口。

那他们之间时如何协作的呢

javascript主线程

程序启动,V8引擎会首先解析javascript代码,通过Node bindings来调用C/C++库。执行到当前事件时,会把事件放在调用堆栈(stack和heap)处理(可以理解为放进一个工作空间,如上图),在堆栈中的任何I/O请求都会交给libuv来处理,libuv维持一个线程池,里面是一些工作线程(如下图),请求会调用这些线程来完成任务,这些线程则调用底层的C/C++库。完成时,libuv再把结果返回事件队列等待主线程执行。在此期间,主线程继续执行其他任务。

node 执行特性

单线程、非阻塞型I/O

单线程的意思就是只在一个线程上运行javascript。首先,javascript 在浏览器端是单线程的,这是为了避免多线程产生任务冲突的情况;其次,java和PHP这类多线程后端语言,为避免同步I/O阻塞,每处理一个连接都会产生一个新线程,这样的话在遇到大量并发请求时就会受到物理内存的限制。node 延续了浏览器端单线程javascript,只用一个主线程执行javascript,不断循环遍历事件队列,执行事件。事实上,主线程发出的I/O请求,都会交给其他线程去完成,其他线程完成后悔返回结果放到事件队列。在此期间,主线程会继续执行其他任务,也就是在交给libuv后直接返回,继续执行下面的任务,主线程只负责循环执行事件队列,因此这种模式称为非阻塞型I/O,性能很好,适用于处理大量并发请求,还能简化开发。

事件驱动机制

还是跟浏览器的差不多。总的来说就是,浏览器端把鼠标点击、键盘按键等定义为事件,而node把网络请求、I/O操作等也看作事件,严格来说,一切动作都是事件,这就是事件驱动的思想。在程序启动时,便进入事件循环,不断遍历执行事件队列中产生的事件,而在执行过程中,又会产生新的事件,因此称为事件循环。主线程执行事件时,遇到麻烦的I/O请求会交给libuv来调度其他工作线程来帮忙,忙完后就会形成事件返回结果到事件队列等待主线程处理。在此期间,主线程会继续执行其他任务。

mbp 曾经做过一个巧妙的比喻,把 Node.js 看成一家餐厅。我在此借用下他的例子,稍作修改来阐述下 Node.js 的执行情况:

把 Node.js 应用程序想象成一家星巴克,一个训练有素的前台服务生(唯一的主线程)在柜台前接受订单。当很多顾客同时光临的时候,他们排队(进入事件队列)等候接待;每当服务生接待一位顾客,服务生会把订单告知给经理(libuv),经理安排相应的专职人员去烹制咖啡(工作线程或者系统特性)。这个专职人员会使用不同的原料和咖啡机(底层 C/C++ 组件)按订单要求制作咖啡或甜点,通常会有四个这样的专职人员保持在岗待命(线程池),高峰期的时候也可以安排更多(不过需要在一早就安排人员来上班,而不能中午临时通知)。服务生把订单转交给经理之后不需要等着咖啡制作完成,而是直接开始接待下一位顾客(事件循环放进调用堆栈的另一个事件),你可以把当前调用堆栈里的事件看成是站在柜台前正在接受服务的顾客。

当咖啡完成时,会被发送到顾客队列的最后位置,等它移动到柜台前服务生会叫相应顾客的名字,顾客就来取走咖啡(最后这部分在真实生活中听起来有点怪,不过你从程序执行的角度理解就比较合乎情理了)。

​ ——By Amio

如果你想进一步了解javascript 事件驱动机制,推荐深入理解 javascript 事件循环机制

node 模块

node 模块机制是CommonJs 的实现。起初,javascript 标准一片混沌,并没有其他成熟语言(例如C++)的模块机制、标准库、接口等,为了让javascript 具备开发大型应用的能力,为了让 javascript 能在后端运行,CommonJS 就制定了javascript 模块规范。node 借鉴了这个规范,让javascript 以模块形式组织起来。模块机制是一个成熟语言必备的,一个模块代表一个功能的封装,它就像搭积木一样,不同模块可以衔接在一块,使语言具有极强的可扩展型。node 模块机制同时制定了模块规范,能让全球的开发者都可以在node官网上传自己的包。此外,node 社区又实现了node 包管理器npm,使用npm可以轻松管理各种包。

node 的模块分为核心模块和用户模块,前者是底层的、自带的,后者是第三方。

核心模块有Global(全局对象)、Http、fs(文件系统)、Buffer、Stream、Events、URL、path等,这些模块提供了后端服务的基本功能,都提供自己关注功能的API。

在使用模块时,require 即可。但在require背后,node 有一套寻找模块的机制:

node require机制

从上图可以看到,node 优先从缓存区读取,缓存区有直接读取,没有则加载并缓存,这样做不用一遍一遍去找了,非常高效。node 在缓存区没有发现模块时,会分析require 的路径和文件后缀,node 有个模块路径的查找策略,我们可以在名为module_paths 的js文件里console.log(module.paths)然后node module_paths.js运行来间接查看node 寻找文件模块的具体文件的方式:

[ '/home/username/nodeProject/node_modules',
 '/home/username/node_modules',
 '/home/node_modules',
 '/node_modules' ] //Linux下的数组输出(/home/username因电脑不同而异)

[ 'c:\\nodeProject\\node_modules', 'c:\\node_modules' ] //Windows

也就是按照下面的顺序:

  • 当前文件目录下的node_modules目录。
  • 父目录下的node_modules目录。
  • 父目录的父目录下的node_modules目录。
  • 沿路径向上逐级递归,直到根目录下的node_modules目录。

这些顺序都是在查找缓存之后的。

在找到模块后,node 将在引入之前对这个模块进行编译执行,编译成功后会缓存,执行的结果会返回给调用者。

简单应用

有了node 自带核心模块的基础功能,就可以进一步封装更强大、容易操作的功能了,就像jQuery 对于javascript 基础API 一样,node 社区也诞生了像 Express、KOA等框架来构建node.js程序

node.js开发框架

这些框架的详情移步2017 Node.js 开发框架比较

另外,node 还可以连接MySQL,MangoDB进行数据库操作。

下面是使用express 脚手架生成的基本 node应用结构:

.
├── app.js            //程序入口
├── bin
│   └── www           //二机制文件
├── package.json      //项目配置文件
├── public
│   ├── images        
│   ├── javascripts
│   └── stylesheets
│       └── style.css
├── routes
│   ├── index.js      //路由文件入口
│   └── users.js
└── views
    ├── error.jade    //界面模板
    ├── index.jade
    └── layout.jade

现在使用 node作为后端语言通常都要配合类库和框架使用。

node 的单线程、非阻塞型特点让它非常适合高并发的应用,适合处理大量重复的、简单的逻辑,适合构建Rest/JSON API服务;同时,也正是因为这些特性,node 不适合CPU使用率较重、IO使用率较轻的偏计算应用。缺点是因为单线程,一个进程挂就全挂了,可靠性低,但这是可以避免的。node 更多的应用是在前端、中间件、前后端分离等。

由于 node 的诸多优点,现在越来越多大公司开始使用node、深度使用node。

总结

node 的核心概念、思想远不止这么多,应用更是多了去了,无奈本人水平有限,只能说个浅层,还有很多像进程管理、异步编程、异常调试、部署、性能调优、与集群、CDN协调等都值得深入探索一下。无论如何,node 是让javascript 迈向企业级开发语言重要的一步(也许已经是了),前端工程师从未像现在这样的powerful,能做的事情越来越多,所能涉及的领域也越来越多。前端这行越来越令人兴奋了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容

  • Node.js是目前非常火热的技术,但是它的诞生经历却很奇特。 众所周知,在Netscape设计出JavaScri...
    w_zhuan阅读 3,612评论 2 41
  • Node.js是目前非常火热的技术,但是它的诞生经历却很奇特。 众所周知,在Netscape设计出JavaScri...
    Myselfyan阅读 4,069评论 2 58
  • 人有许多路可以选,可以走。
    马不蹄儿阅读 347评论 0 0
  • “这么说,你老婆是南方人咯?”我紧接着问道。 “恩。”对面的大叔腼腆的笑笑,被长年累月的油烟熏得油光滑亮的脸上绽开...
    美好的花想容阅读 462评论 0 0
  • 黑夜中响起璀璨 恍若烟火明亮 一闪一响,此起彼伏 单调而又凄凉 这绝不是人间的光 那么刺眼又无情 灵魂被收割 发出...
    淺笑大大阅读 299评论 0 0