模块
nodejs采用的是commonjs的模块机制,commonjs模块的定义很简单,主要分为模块引用require,模块定义export和模块标示(path)三个部分。
node模块实现
node的模块分为两类:
核心模块:在node源码编译过程中,编译而成的二进制文件。node在启动时直接把核心模块加载进入内存,所以其加载速度最快。
文件模块:运行时动态加载,需要完整的路径分析,文件定位,编译执行,速度比核心模块慢。
在node模块通过require()方法加载,一律采用缓存优先方式,并且核心模块缓存检查优先于文件模块。
路径分析
路径分析主要分为三类:核心模块,路径形式文件模块和自定义模块。核心模块加载最快,自定义模块加载最慢。
文件定位
require
require是一个方法,用于接收模块标识符,引入一个模块API到当前上下文环境中
module.exports 与 exports
module对象是模块本身,exports是module的属性,用于导出当前模块的方法或者变量。
热更新
热更新的最主要原因在于,nodejs的模块加载机制,require是默认优先从缓存加载,只要缓存存在,就不会再重新去加载文件,避免了二次引入的开销。热更新的方法也就有两种:
- 清空模块的cash,当重新引用模块时便会从从文件再次加载。不过这种方法可能存在文件,就是之前在模块中保存的上下文环境,临时变量会被清空已经旧代码可能保存了旧的cash对象的引用。这个问题可能从最初的时候时候就需要考虑到。不过如果是普通的json配置文件,通过这种方法引用不会有任何问题。
- 存储数据库,继续需要频繁的更新,可以通过db数据库去存储这些信息。
上下文
对于 Node.js 而言, 正常情况下只有一个上下文, 甚至于内置的很多方面例如 require
的实现只是在启动的时候运行了内置的函数.
每个单独的 .js 文件并不意味着单独的上下文, 在某个 .js 文件中污染了全局的作用域一样能影响到其他的地方.
而目前的 Node.js 将 VM 的接口暴露了出来, 可以让你自己创建一个新的 js 上下文, 这一点上跟前端 js 还是区别挺大的. 在执行外部代码的时候, 通过创建新的上下文沙盒 (sandbox) 可以避免上下文被污染:
'use strict';
const vm = require('vm');
let code =
`(function(require) {
const http = require('http');
http.createServer( (request, response) => {
response.writeHead(200, {'Content-Type': 'text/plain'});
response.end('Hello World\\n');
}).listen(8124);
console.log('Server running at http://127.0.0.1:8124/');
})`;
vm.runInThisContext(code)(require);
这种执行方式与 eval 和 Function 有明显的区别. 关于 VM 更多的一些接口可以先阅读官方文档 VM (虚拟机)
包管理
常用的就是npm, cnpm,yarn等
npm因为资源问题,经常下载很慢或者下载失败
cnpm之前用过,在windows使用了快捷方式的链接,在通过npm执行的时候,容易找不到module
yarn刚开始接触