成为自信的node.js开发者(一)

这个博客是我最近整理了过去的文章。

适合阅读的同学

想更进一步深入理解node的同学,如果你已经稍微了解一点点node, 可以用node做一些小demo,并且想更深一步的了解,希望这篇文章可以帮助到你。

不太适合阅读的同学

  1. 不太熟悉基本的javascript 语法,比如说回调函数

  2. 对node有深入理解的同学,比如说,可以清晰的说出event-loop

Node 架构——v8、libuv

第一部分,我们先了解一下node的结构,对node先有一个整体上的认识。只有这样,我们才能编写出更加高性能的代码,在遇到问题时,也知道解决的思路。

先来看一张图表:

最上面是我们编写的node.js的代码,当我们执行node index.js的命令时,我们是触发了一个node的程序,和其他的javascript的项目,比如说前端的h5项目一样,该node程序需要有其他的依赖,其中最主要的两个依赖是 v8libuv

  1. v8是 google 开源的引擎,目的是在浏览器世界外可以运行javascript的代码。

  2. libuv 是c++ 开源的项目,最初就是专门为node设计,目的是给node和操作系统交互的能力,比如说网络, 操作文件。

node在可见的未来仍将继续使用v8, 但是微软edge浏览器的chakra(读法:渣坷垃)引擎也是一个强有力的竞争者。github.com/nodejs/node… 这个项目是如何让node如何跑在chakras 引擎上

v8 引擎

我们现在知道了,node 使用 v8 用来执行javascript 代码,这意味着,node中所支持的javascript的特性,是由 v8 引擎所决定的。

V8引擎支持的 javascript 特性被划分为三个不同的group: ShippingStagedIn Progress

默认情况下Shipping group的特性可以直接使用,Staged group的特性需要使用--harmony选项来开启。如下所示:

➜ node -v
v7.9.0
➜ node -p 'process.versions.v8'
5.5.372.43
➜ node -p "'Node'.padEnd(8, '*')"    // 默认是不支持的
[eval]:1
'Node'.padEnd(8, '*')^TypeError: "Node".padEnd is not a functionat [eval]:1:8at ContextifyScript.Script.runInThisContext (vm.js:23:33)at Object.runInThisContext (vm.js:95:38)at Object. ([eval]-wrapper:6:22)at Module._compile (module.js:571:32)at evalScript (bootstrap_node.js:387:27)at run (bootstrap_node.js:120:11)at run (bootstrap_node.js:423:7)at startup (bootstrap_node.js:119:9)at bootstrap_node.js:538:3
➜ node --harmony -p "'Node'.padEnd(8, '*')"  // 通过--harmony
Node****
复制代码

In Progress group的feature不稳定,但你也可以使用特定的flag来开启,通过 node --v8-options 命令可以查看,通过grep 命令去查找in progress,如下:

➜ node --v8-options | grep "in progress"--harmony_array_prototype_values (enable "harmony Array.prototype.values" (in progress))--harmony_function_sent (enable "harmony function.sent" (in progress))--harmony_sharedarraybuffer (enable "harmony sharedarraybuffer" (in progress))--harmony_simd (enable "harmony simd" (in progress))--harmony_do_expressions (enable "harmony do-expressions" (in progress))--harmony_restrictive_generators (enable "harmony restrictions on generator declarations" (in progress))--harmony_regexp_named_captures (enable "harmony regexp named captures" (in progress))--harmony_regexp_property (enable "harmony unicode regexp property classes" (in progress))--harmony_for_in (enable "harmony for-in syntax" (in progress))--harmony_trailing_commas (enable "harmony trailing commas in function parameter lists" (in progress))--harmony_class_fields (enable "harmony public fields in class literals" (in progress))
复制代码

比如说,上面打印出来的倒数第二行-- harmony_trailing_commas 可以支持函数传参尾逗号:

node -p 'function tc(a,b,) {}'   // 会报错,因为最后一个逗号
=========================
node --harmony_trailing_commas -p 'function tc(a,b,) {}'   //不会报错
复制代码

libuv

  1. libuv 提供了和操作系统交互的能力,比如说操作文件,网络等等,并且磨平了操作系统的差异。

  2. node还使用libuv来处理异步操作,比如非阻塞IO(file system/TCP socket/child process)。当异步操作完成时,node通常需要调用回调函数,当调用回调函数时,node会把控制权交给V8引擎。 当回调函数执行完毕,控制权从v8引擎重新回到node.

    v8 引擎是单线程的,当v8引擎获得控制权的时候,node 只能等待v8 引擎操作完成。

    这让node没有死锁,竞争的概念。

  3. libuv 包含一个线程池,从操作系统的层面来做那些不能被异步做的事情

  4. libuv 给node 提供了 event-loop, 会在第二节介绍

其他依赖

除了v8引擎和 libuv, node 还有其他的一些比较重要的依赖。

http-parser 用来解析http内容的

c-ares 是用来支持异步的DNS 查询的

openSSL 常用在 tlscrypto 的包中,提供了加密的方法

zlib 是用来压缩和解压的

node REPL

你可以在terminal里面执行node来启动CLI,如下所示,REPL十分方便

例如,你定义一个array,当你arr.然后tab-tab(tab两次),array自身的方法会显示出来

➜ node
> var arr = [];
undefined
> arr.
arr.toString              arr.valueOf
arr.concat                arr.copyWithin         arr.entries               arr.every              arr.fill                  arr.filter
arr.find                  arr.findIndex          arr.forEach               arr.includes           arr.indexOf               arr.join
arr.keys                  arr.lastIndexOf        arr.length                arr.map                arr.pop                   arr.push
arr.reduce                arr.reduceRight        arr.reverse               arr.shift              arr.slice                 arr.some
arr.sort                  arr.splice             arr.unshift
复制代码

你也可以输入.help,然后可以看到各种快捷键如下:

> .help
.break    Sometimes you get stuck, this gets you out
.clear    Alias for .break
.editor   Enter editor mode
.exit     Exit the repl
.help     Print this help message
.load     Load JS from a file into the REPL session
.save     Save all evaluated commands in this REPL session to a file
复制代码

你还可以用_(underscore)来得到上次evaluated的值:

> 3 - 2
1
> _
1
> 3 < 2
false
> _
false
复制代码

你还可以自定义REPL选项,如下,你自定义repl.js并选择忽视undefined,这样output里面就不会有undefined输出,同时你还可以预先加载你需要的library比如lodash

// repl.js
let repl = require('repl');
let r = repl.start({ ignoreUndefined: true  });
r.context.lodash = require('lodash');
复制代码
➜ node ~/repl.js
> var i = 2;
> 
>
复制代码

你可以用下面的command来查看更多的选项 node --help | less

-p, --print     evaluate script and print result-c, --check     syntax check script without executing-r, --require   module to preload (option can be repeated)
复制代码

例如,node -c bad-syntax.js可以用来检查语法错误, node -p 'os.cpus()'可以用来执行script并输出结果,你还可以传入参数,如下所示

➜ node -p 'process.argv.slice(1)' test 666
[ 'test', '666' ]
复制代码

node -r babel-core/register可以用来预加载,相当于require('babel-core/register')

global 中的 process 和 buffer

global相当于浏览器里面的window,你可以global.a = 1;这样a就是全局变量,但一般不推荐这样做

global 对象身上有两个属性特别重要: processbuffer

process

processapplicationrunning env之间的桥梁,可以得到运行环境相关信息,如下所示:

> process.
process.arch
process.argv
process.argv0                       process.assert                      process.binding                     
process.chdir
process.config                      process.cpuUsage                    
process.cwd                         process.debugPort
process.dlopen                      process.emitWarning                 
process.env                         process.execArgv
process.execPath                    
process.exit                        process.features                    process.getegid
process.geteuid                     process.getgid                      process.getgroups                   process.getuid
process.hrtime                      process.initgroups                  
process.kill                        process.memoryUsage
process.moduleLoadList              process.nextTick                    process.openStdin                   
process.pid
process.platform                    process.reallyExit                  process.release                     process.setegid
process.seteuid                     process.setgid                      process.setgroups                   process.setuid
process.stderr                      process.stdin                       process.stdout                      
process.title
process.umask                       process.uptime                      process.version                     process.versions
process._events                     process._maxListeners               process.addListener                 process.domain
process.emit                        process.eventNames                  process.getMaxListeners             process.listenerCount
process.listeners                   
process.on                          
process.once                        process.prependListener
process.prependOnceListener         process.removeAllListeners          process.removeListener              process.setMaxListeners\
复制代码

process.versions 非常有用:

process.env 提供了当前环境的一些信息

建议从 process.env 中只读,因为改了也没有用。

同时,process也是一个event emitter,例如:

process.on('exit', code => {// 并不能阻止node进程退出console.log(code)
})process.on('uncaughtException', err => {console.error(err)process.exit(1)
})
复制代码
  1. 在process 的事件处理函数中,我们只能执行同步的方法,而不能使用event_loop,

  2. exituncaughtException 的区别。如果uncaughtException 注册了事件,则node遇到错误并不会退出,也就是说,不会触发exit 事件。这会让node的执行变的不可预测。证明如下:

    process.on('exit', (code) => {console.log('ssss')    
    })
    process.on('uncaughtException', (err) => {console.error(err);
    })
    // keep the event loop busy
    process.stdin.resume()// 在这里触发了bug
    console.logg()
    复制代码

    上面的代码即使遇到了错误也不会退出执行,exit 事件处理函数并不会触发。所以需要我们手动触发 process.exit(1) 才可以。

buffer

buffer 也是 global 对象中的一个属性,主要用来处理二进制流buffer 本质上是一段内存片段,是放在v8引擎的堆的外面。

我们可以在buffer 这个内存中存放数据。

buffer读取数据时,我们必须指定encoding, 因此从 filessockets 中读取数据时,如果不指定encoding, 我们会得到一个 buffer 对象。

一旦buffer 被创建,就不能修改大小

buffer 在处理读取文件,网络数据流的时候非常有用

创建buffer的三种方式:

  1. Buffer.alloc(2)

    在内存中划分出固定的大小

  2. Buffer.allocUnsafe(8)

    没有指定具体的数据,可能会包含老的数据和敏感的数据,需要被正确的『填充』

  3. Buffer.from()

buffer的方法

和数组类似,但是不同。比如说 slice 方法截取出来的新buffer 和 老的buffer是共享同一个内存。

stringDecode

当转变二进制数据流的时候,toString() 不如使用 stringDecode 模块,因为该模块可以处理不完整的数据呢。

Require() 的背后

如果想深入了解node, 必须要深入了解 require 方法。

涉及到两个核心模块——require 方法(在grobal对象上,但是每一个模块都有自己的require 方法) 和 Module 模块 (同样在grobal对象上,用来管理模块的)

require 分为几步

当我们require一个module时,整个过程有五个步骤:

Resolving 找到module的绝对文件路径

Loading 将文件内容加载到内存

Wrapping 给每个module创造一个private scope并确保require对每个module来说是local变量

Evaluating VM执行module代码

Caching 缓存module以备下次使用

module 对象

Module {id: '.',exports: {},parent: undefined,filename: '/Users/xxx/lib/find.js',loaded: false,children: [],paths: [ '/Users/xxx/lib/node_modules','/Users/xxx/node_modules','/Users/node_modules','/node_modules' ] }
复制代码

在Module对象里面,id 是module的identity,通常它的值是module文件的全路径,除非是root,这时它的值是.(dot)

filename 是文件的路径

paths 从当前路径开始,往上一直到根路径

require.resolve 和require一样,但是它不会加载文件,只是resolve

模块不一定是文件

  1. 可以是文件,比如说 node_module/find-me.js

  2. 可以是目录带index.js,比如说 node_module/find-me/index.js

  3. 可以是目录带package.json, 比如说node_module/find-me/main.js

    {"name": "find-me","main": "start.js"
    }
    复制代码

exports 属性

exportsmodule 上一个特殊的属性,我们放入它的任何变量都可以在require时得到 。

loaded

Module对象的loaded属性会保持false,直到所有content都被加载

因此,exports 不能放在的异步的setImmediate

循环引用

例如A require B,B require A

JSON 文件 和 c++ Addon 文件

Node会首先查找.js文件,再查找.json文件,最后.node文件 比如说,在主文件中,引入.json 文件

// 在主文件中
let mock = require('mockData.json')
console.log(mock)
复制代码

mockData.json 文件中,不需要导出什么,直接写json格式的即可

{"a": "abc","b": "abc",
}
复制代码

如果node找不到 .js , .json 文件,就会找.node 文件,会把.node 文件作为一个编译好的addon(插件) module。那么 .node 文件是从哪里来的呢?

  1. 先有一个 hello.cc 文件,是用 c++ 代码写的

  2. 再有一个 binding.gyp, 相当于的编译的配置文件,里面是json 格式的配置项, 如下面所示:

    {"targets": [{"target_name": "addon","sources": [ "hello.cc" ]}]
    }
    复制代码
  3. 安装 npm install node-gyp -g , node 和 npm 自带的那个不是给开发者用的,而是需要重新安装一个

  4. node-gyp configure 根据平台生成项目,再执行node-gyp build 生成 .node 文件,可以在 js的代码中直接引用使用了。

    你可以通过require.extensions来查看Node支持的文件扩展名:

> require.extensions
{ '.js': [Function], '.json': [Function], '.node': [Function] }
复制代码

上面的代码中,对于 .js 文件,是直接编译引入,对于.json 文件,是使用了JSON.parse 方法,对于 .node 文件,是使用了 process.dlopen() 方法。

包裹模块

exports.id = 1;   // 对的exports = {id: 1,        // 错的
}module.exports = {id: 1        // 对的
}
复制代码

上面的代码中,为什么exportsmodule.exports 有区别?

原因是,node 引入一个模块代码后,node 会给这些代码外面包裹上一层方法,这个方法是module 模块的wrapper 方法:

> require('module').wrapper
>[ '(function (exports, require, module, __filename, __dirname) { ','\n});' ]
复制代码

这个方法接受5个参数: exports, require, module, __filename, __dirname

这个方法,让 exports, require, module 看起来是全局变量,但其实是每个文件所独有的。

exportsmodule 对象的module.exports 方法的引用,相当于 let exports s = module.exports, 如果让 exports = {} 等于让 exports 变量改写了引用

缓存模块

当第二次引入同一个文件的时候,将会走了缓存。

console.log(require.cache)
delete require.cache['/User/sss/sss/cache.js']
复制代码

下一期我们再见~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/275857.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql读写分离和分布式_MySQL主从复制与读写分离

MySQL主从复制(Master-Slave)与读写分离(MySQL-Proxy)实践Mysql作为目前世界上使用最广泛的免费数据库&#xff0c;相信所有从事系统运维的工程师都一定接触过。但在实际的生产环境中&#xff0c;由单台Mysql作为独立的数据库是完全不能满足实际需求的&#xff0c;无论是在安全…

ux和ui_UI和UX设计师的10种软技能

ux和ui重点 (Top highlight)As designers, whether it be UI, UX, or Product Design, we tend to direct our focus and energy on developing and mastering tangible skills.作为设计师&#xff0c;无论是UI&#xff0c;UX还是产品设计&#xff0c;我们都将重点和精力放在开…

SQLServer中批量插入数据方式的性能对比 (转)

转自&#xff1a;http://www.cnblogs.com/wlb/archive/2010/03/02/1676136.html 昨天下午快下班的时候&#xff0c;无意中听到公司两位同事在探讨批量向数据库插入数据的性能优化问题&#xff0c;顿时来了兴趣&#xff0c;把自己的想法向两位同事说了一下&#xff0c;于是有了本…

VueConf China 2021 《Vue3生态进展-尤雨溪》 Reaction

大家好&#xff0c;我是若川。今天分享昨天Vueconf的一篇文章&#xff0c;来了解下Vue的生态进展。另外今晚7点&#xff0c;Vuebeijing社区邀请了尤大会在视频号直播&#xff0c;可以加我微信 ruochuan12&#xff0c;告诉观看地址提前预约。点击下方卡片关注我、加个星标&#…

Plsql运行mysql脚本_oracle中PLSQL语句

1.set autot off 禁止使用autotrace命令 set autot on 这个命令包括exp 和 stat(执行语句、生成explain plan、生成统计信息) set autot trace 不执行sql语句&#xff0c;但(生成explain plan、生成统计信息) set autot trace exp stat 与上句同 set autot trace st1.set autot…

2019年,你需要关注这些Node API和Web框架

对于Node.js框架和开源软件来说&#xff0c;2018年是非常有趣的一年。开发者社区讨论了企业赞助对开源项目的作用以及如何维护那些没有经济支持却有数百万人使用的项目。同样&#xff0c;安全问题也得到了极大关注&#xff0c;一些流行的Node/JS软件包被劫持&#xff0c;Github…

ai创造了哪些职业_关于创造职业的思考

ai创造了哪些职业When I was growing up, the idea of a creative career wasn’t an option.当我长大时&#xff0c;创意事业的想法不是一个选择。 I had enjoyed doodling, arts and crafts as a kid, so as I grew up, it was a natural transition into Photoshop and lat…

Windows Mobile,用C#更改网络连接(SSID、IP Address、Subnet Mask、Gatew... (转)

前几天在做一个改变PDA无线网络连接的SSID和IP的功能是发现了一个好东西OpenNETCF Framework使用OpenNETCF.Net包&#xff0c;实现了任意改变PDA无线网络连接的功能。并且不需要Reset PDA。现在正在做一个IP Manager For Windows Mobile的小程序。实现搜索当前网卡可见的SSID、…

一文读懂vuex4源码,原来provide/inject就是妙用了原型链?

1. 前言你好&#xff0c;我是若川&#xff0c;欢迎加我微信ruochuan12&#xff0c;加群长期交流学习。这是学习源码整体架构系列 之 vuex4 源码&#xff08;第十篇&#xff09;。学习源码整体架构系列文章(有哪些必看的JS库)&#xff1a;jQuery、underscore、lodash、sentry、v…

Spring4.3x教程之一IOCDI

SpringIOC也称为DI&#xff0c;对属性内容的注入可以通过属性的setXXX方法进行也可以通过构造方法进行&#xff0c;当然还可以使用工厂模式进行属性内容的注入。 什么是DI&#xff1f;什么是IOC&#xff1f; DI&#xff1a;Dependency Injection依赖注入 其实一个类中的属性就是…

战神4 幕后花絮 概念艺术_幕后花絮:品牌更新的背后

战神4 幕后花絮 概念艺术Under the Hood gives you an inside look at different parts of Waze — straight from the people working on them every day.在引擎盖下&#xff0c;您可以深入了解Waze的不同部分-直接来自每天进行工作的人员。 Traffic is the worst. It makes …

C#日期控件(js版)

js 脚本代码: <script type"text/javascript"> //---------------------------------------------------------------------------- //---------------------------------------------------------------------------- // 这是一个日历 Javascript 页…

python第三周测试_python第三周小测

1.读取一个文件&#xff0c;显示除了井号(#)开头的行意外的所有行# -*- coding: utf-8 -*-"""Created on Tue May 28 09:37:08 2019author: Omega_Sendoh"""#打开文件f open("install-sh","r")#读取文件的所有行&#xff0…

「Vueconf」探索 Vue3 中 的 JSX

大家好&#xff0c;我是若川。今天再分享 Vueconf 的一篇文章。另外 Vueconf 主办方提供的录播链接是&#xff1a; https://www.bilibili.com/read/mobile?id11408693&#xff0c;感兴趣可以复制观看。点击下方卡片关注我、加个星标。学习源码整体架构系列、年度总结、JS基础…

设计模式--享元模式实现C++

/********************************* *设计模式--享元模式实现 *C语言 *Author&#xff1a;WangYong *Blog:http://www.cnblogs.com/newwy ********************************/ #include <iostream> #include <cassert> #include <vector> #include <strin…

安卓加载asset中的json文件_Android解析Asset目录下的json文件

在app module中的src/main/assets目录下我们准备了两个json文件&#xff1a;destination.json如下&#xff1a;{"main/tabs/sofa": {"isFragment": true,"asStarter": false,"needLogin": false,"pageUrl": "main/tabs…

一文搞懂 Promise、Genarator、 Async 三者的区别和联系

非985/211大学毕业&#xff0c;软件工程专业&#xff0c;前端&#xff0c;坐标&#xff1a;北京工作三年多&#xff0c;第一家人数 30 多人的创业公司&#xff0c;1 年多。第二家属于前端技术不错的公司&#xff0c;2 年多。01我是一个喜欢在技术领域“折腾”的人&#xff0c;技…

闭包,sync使用细节

代码 先看代码如下&#xff1a; func main() {var a []intfor i : 0; i < 100; i {go func() {a append(a, i)}()}time.Sleep(2 * time.Second)fmt.Println(a) } 这段测试代码是想要一个元素为0到100的切片&#xff0c;但是这一小段代码隐藏了很多的问题。 闭包函数 先看这…

dynamic 仪表板_仪表板完成百万美元交易

dynamic 仪表板问题 (The Problem) Anybody dealing with tech products and data-focused services runs into the same fundamental problem: what you do is technical but non-technical people control the budget. In other words:任何处理高科技产品和以数据为中心的服务…

checkStyle -- 代码风格一致

download page: http://sourceforge.net/project/showfiles.php?group_id80344&package_id107587 转载于:https://www.cnblogs.com/xuqiang/archive/2010/10/26/1953431.html