Skip to content

Commit ac30eeb

Browse files
module
1 parent 0c16c73 commit ac30eeb

File tree

22 files changed

+345
-0
lines changed

22 files changed

+345
-0
lines changed

模块/module.md

+101
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,101 @@
1+
## 模块机制
2+
3+
### 循环引用问题
4+
5+
```js
6+
// ---------- circular/a.js -----------
7+
8+
console.log(`a begin`);
9+
exports.done = false;
10+
11+
const b = require('./b');
12+
console.log(`a load b.js, b.done = ${b.done}`);
13+
14+
exports.done = true;
15+
console.log(`b end`);
16+
17+
// ---------- circular/b.js -----------
18+
console.log(`b.js begin`);
19+
exports.done = false;
20+
21+
const a = require('./a');
22+
console.log(`b load a.js, a.done = ${a.done}`);
23+
24+
exports.done = true;
25+
console.log(`b end`);
26+
27+
28+
// ---------- index.js -----------
29+
console.log(`circular load a.js and b.js begin`);
30+
const a = require('./circular/a');
31+
const b = require('./circular/b');
32+
console.log(`index.js a.done = ${a.done}, b.done = ${b.done}`);
33+
console.log(`circular load a.js and b.js end`);
34+
35+
// ---------- console -----------
36+
/*
37+
circular load a.js and b.js begin
38+
a begin
39+
b.js begin
40+
b load a.js, a.done = false
41+
b end
42+
a load b.js, b.done = true
43+
b end
44+
index.js a.done = true, b.done = true
45+
circular load a.js and b.js end
46+
* */
47+
```
48+
加载流程如下
49+
50+
* index.js 加载 a.js,
51+
* a 尝试加载 b.js
52+
* b.js 这时候尝试加载 a.js
53+
* 为防止死循环, 此时, b.js 只能拿到一个 a.js 的未完成的副本进行使用
54+
* b.js 完成加载后 a.js 拿到 b.js 的 exports,
55+
* a.js 再完成剩余的加载
56+
* 最终, 在 index.js 中 a.js 和 b.js 均完成加载
57+
58+
需注意:循环过程中,b.js 中拿到的 a.js 是不完整的,部分属性方法可能无法使用。虽然循环引用不会导致死循环,但也应该避免。
59+
60+
61+
### 全局变量泄露问题
62+
63+
```js
64+
// ---------- global/leakGlobalVar.js -----------
65+
leak = 'this is a leak global variable'; // 可启用严格模式避免
66+
module.exports = {
67+
name: 'leakGlobalVar.js'
68+
};
69+
70+
// ----------index.js -----------
71+
const leakGlobalVar = require('./global/leakGlobalVar');
72+
console.log(`module: ${JSON.stringify(leakGlobalVar)}, leak: ${leak}`); // module: {"name":"leakGlobalVar.js"}, leak: this is a leak global variable
73+
```
74+
75+
模块中泄露的全局变量可在其他模块中拿到, 无意识的变量泄露可通过启用严格模式避免
76+
77+
```js
78+
// ---------- global/vm.js -----------
79+
const vm = require('vm');
80+
let code = `(
81+
function() {
82+
leakInVm = 'this is a leak object in vm';
83+
return {
84+
name: 'global/vm.js'
85+
}
86+
}
87+
)`;
88+
const vmFn = vm.runInThisContext(code);
89+
module.exports = vmFn();
90+
91+
// ---------- index.js -----------
92+
93+
const vm1 = require('./global/vm');
94+
console.log(vm1); // { name: 'global/vm.js' }
95+
console.log(leakInVm); // this is a leak object in vm
96+
97+
```
98+
[vm.runInThisContext() compiles code, runs it within the context of the current global and returns the result. Running code does not have access to local scope, but does have access to the current global object.](https://nodejs.org/docs/latest-v8.x/api/vm.html#vm_vm_runinthiscontext_code_options)
99+
100+
101+

模块/module/circular/a.js

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
console.log(`a begin`);
2+
exports.done = false;
3+
4+
const b = require('./b');
5+
console.log(`a load b.js, b.done = ${b.done}`);
6+
7+
exports.done = true;
8+
console.log(`b end`);

模块/module/circular/b.js

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
console.log(`b.js begin`);
2+
exports.done = false;
3+
4+
const a = require('./a');
5+
console.log(`b load a.js, a.done = ${a.done}`);
6+
7+
exports.done = true;
8+
console.log(`b end`);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
name: "dirModule1/index.js"
3+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"name": "dirModule1",
3+
"version": "1.0.0",
4+
"main": "index.js",
5+
"dependencies": {
6+
7+
}
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
name: "dirModule2/lib/index.js"
3+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"name": "dirModule2",
3+
"version": "1.0.0",
4+
"main": "lib",
5+
"dependencies": {
6+
7+
}
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
name: "dirModule3/index.js"
3+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "dirModule3/index.json"
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
{
2+
"name": "dirModule3",
3+
"version": "1.0.0",
4+
"dependencies": {
5+
6+
}
7+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "dirModule4/index.json"
3+
}

模块/module/fileModule/file1

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
"name": "file1"
3+
};

模块/module/fileModule/file1.js

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
"name": "file1.js"
3+
};

模块/module/fileModule/file2.js

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
module.exports = {
2+
"name": "file2.js"
3+
};

模块/module/fileModule/file2.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "file2.json"
3+
}

模块/module/fileModule/file3.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "file3.json"
3+
}

模块/module/global/leakGlobalVar.js

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
2+
leak = 'this is a leak global variable'; // 可启用严格模式避免
3+
4+
module.exports = {
5+
name: 'leakGlobalVar.js'
6+
};

模块/module/global/vm.js

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
const vm = require('vm');
2+
3+
let code = `(
4+
function() {
5+
leakInVm = 'this is a leak object in vm';
6+
return {
7+
name: 'global/vm.js'
8+
}
9+
}
10+
)`;
11+
12+
const vmFn = vm.runInThisContext(code);
13+
module.exports = vmFn();
14+
15+

模块/module/index.js

+116
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,116 @@
1+
2+
// 模块
3+
const circle = require('./shape/circle');
4+
console.log(`半径为 3 的圆的面积为: ${circle.area(3)}`); // 半径为 3 的圆的面积为: 28.274333882308138
5+
6+
const Square = require('./shape/square');
7+
const mySquare = new Square(4);
8+
console.log(`mySquare 的面积为: ${mySquare.area()}`); // mySquare 的面积为: 16
9+
10+
// 主模块
11+
const isMain = require.main === module;
12+
if (isMain) console.log(`main module: ${require.main.filename} `);// main module: /.../module/index.js
13+
14+
// 获取模块加载名
15+
const cycleName = require.resolve('./shape/circle');
16+
console.log(`${cycleName}`); // /.../module/index.js
17+
18+
// 模块加载缓存,
19+
const Square1 = require('./shape/square'); // square.js 中的 console.log 只输出一次
20+
const Square2 = require.cache[require.resolve('./shape/square')].exports;
21+
console.log(`Square === Square1: ${Square === Square1}\nSquare === Square2: ${Square === Square2}`); // Square === Square1: true Square === Square2: true
22+
23+
// 循环导入问题
24+
console.log(`circular load a.js and b.js begin`);
25+
const a = require('./circular/a');
26+
const b = require('./circular/b');
27+
console.log(`index.js a.done = ${a.done}, b.done = ${b.done}`);
28+
console.log(`circular load a.js and b.js end`);
29+
/*
30+
circular load a.js and b.js begin
31+
a begin
32+
b.js begin
33+
b load a.js, a.done = false
34+
b end
35+
a load b.js, b.done = true
36+
b end
37+
index.js a.done = true, b.done = true
38+
circular load a.js and b.js end
39+
40+
index.js 加载 a.js 时, a 尝试加载 b.js
41+
但是 b.js 这时候尝试加载 a.js, 为防止死循环, 此时, b.js 只能拿到一个 a.js 的未完成的副本进行使用
42+
b.js 完成加载后 a.js 拿到 b.js 的 exports, a.js 再完成剩余的加载
43+
最终, 在 index.js 中 a.js 和 b.js 均完成加载
44+
* */
45+
46+
47+
// require 加载规则 -> http://nodejs.cn/api/modules.html#modules_all_together
48+
// 文件加载 LOAD_AS_FILE(X)
49+
const file1 = require('./fileModule/file1'); // If X is a file, load X as JavaScript text. STOP,
50+
console.log(`file1 is ${JSON.stringify(file1)}`); // file1 is {"name":"file1"}, 虽然 file1.js 存在, 但是查询到 file1 后停止
51+
52+
const file2 = require('./fileModule/file2'); // If X.js is a file, load X.js as JavaScript text. STOP
53+
console.log(`file2 is ${JSON.stringify(file2)}`); // file2 is {"name":"file2.js"} 虽然 file2.json 存在, 但是查询到 file2.js 停止
54+
55+
const file3 = require('./fileModule/file3'); // If X.json is a file, parse X.json to a JavaScript Object. STOP
56+
console.log(`file3 is ${JSON.stringify(file3)}`); // file3 is {"name":"file3.json"}, .node 文件未制作
57+
// .node 插件待制作, 也会尝试加载 后缀为 .node的文件
58+
59+
60+
// 目录加载, LOAD_AS_DIRECTORY(X) -> LOAD_INDEX(X)
61+
// LOAD_AS_DIRECTORY(X)
62+
// 1. If X/package.json is a file,
63+
// a. Parse X/package.json, and look for "main" field.
64+
// b. let M = X + (json main field)
65+
// c. LOAD_AS_FILE(M)
66+
// d. LOAD_INDEX(M)
67+
// 2. LOAD_INDEX(X)
68+
//
69+
// LOAD_INDEX(X)
70+
// 1. If X/index.js is a file, load X/index.js as JavaScript text. STOP
71+
// 2. If X/index.json is a file, parse X/index.json to a JavaScript object. STOP
72+
// 3. If X/index.node is a file, load X/index.node as binary addon. STOP
73+
74+
75+
const dirModule1 = require('./dirModule/dirModule1'); // package.json -> main -> load file
76+
console.log(`${JSON.stringify(dirModule1)}`); // {"name":"dirModule1/index.js"}
77+
78+
79+
const dirModule2 = require('./dirModule/dirModule2'); // package.json -> main -> LOAD_INDEX(main)
80+
console.log(`${JSON.stringify(dirModule2)}`); // {"name":"dirModule2/lib/index.js"}
81+
82+
const dirModule3 = require('./dirModule/dirModule3'); // package.json -> main? -> LOAD_INDEX(dir)
83+
console.log(`${JSON.stringify(dirModule3)}`); // {"name":"dirModule3/index.js"} 虽然也存在 index.json 但是优先级低
84+
85+
const dirModule4 = require('./dirModule/dirModule4'); // package.json -> main? -> LOAD_INDEX(dir)
86+
console.log(`${JSON.stringify(dirModule4)}`); // {"name":"dirModule4/index.json"}, addon 未准备
87+
88+
// 从 node_modules 目录加载, 此处已 is 模块演示
89+
const is = require('is');
90+
console.log(`${is.string}`); // function (value) {return toStr.call(value) === '[object String]';}
91+
92+
93+
// 从全局目录加载, 如安装的全局模块
94+
// NODE_PATH
95+
// $HOME/.node_modules
96+
// $HOME/.node_libraries
97+
// $PREFIX/lib/node
98+
99+
100+
// 模块封装器
101+
102+
// (function(exports, require, module, __filename, __dirname) {
103+
// // 模块的代码实际上在这里
104+
// });
105+
// 保持顶层变量,作用在模块范围内, 而不是全局对象
106+
// 方便提供看似全局实际上是模块特有的变量
107+
108+
// 子模块可以泄露全局变量吗?
109+
const leakGlobalVar = require('./global/leakGlobalVar');
110+
console.log(`module: ${JSON.stringify(leakGlobalVar)}, leak: ${leak}`); // module: {"name":"leakGlobalVar.js"}, leak: this is a leak global variable
111+
112+
const vm1 = require('./global/vm');
113+
console.log(vm1); // { name: 'global/vm.js' }
114+
console.log(leakInVm); // this is a leak object in vm
115+
// https://nodejs.org/docs/latest-v8.x/api/vm.html#vm_vm_runinthiscontext_code_options
116+
//

模块/module/package.json

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
{
2+
"name": "module-test",
3+
"version": "1.0.0",
4+
"description": "",
5+
"main": "index.js",
6+
"scripts": {
7+
"test": "echo \"Error: no test specified\" && exit 1"
8+
},
9+
"author": "",
10+
"license": "ISC",
11+
"dependencies": {
12+
"is": "^3.3.0"
13+
}
14+
}

模块/module/shape/circle.js

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
const { PI } = Math;
2+
3+
exports.area = r => PI * r ** 2;
4+
exports.circumference = r => 2*PI*r;
5+
6+

模块/module/shape/square.js

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
2+
// exports 仅是 module.exports 的一个引用, 直接修改 exports 的指向是无效的
3+
module.exports = class Square{
4+
constructor(width) {
5+
this.width = width;
6+
}
7+
8+
area() {
9+
return this.width ** 2;
10+
}
11+
};
12+
13+
14+
console.log(`${__filename} load`); // /..../shape/Square.js load
15+
16+
17+
const circle = require('./circle');
18+
console.log(require.cache[require.resolve('./circle')].exports === circle);

0 commit comments

Comments
 (0)