模块的循环加载

如果发生模块的循环加载,即A加载B,B又加载A,则B将加载A的不完整版本。

// a.js
exports.x = 'a1';
console.log('a.js ', require('./b.js').x);
exports.x = 'a2'; // b.js
exports.x = 'b1';
console.log('b.js ', require('./a.js').x);
exports.x = 'b2'; // main.js
console.log('main.js ', require('./a.js').x);
console.log('main.js ', require('./b.js').x);

上面代码是三个JavaScript文件。其中,a.js加载了b.js,而b.js又加载a.js。这时,Node返回a.js的不完整版本,所以执行结果如下。

修改main.js,再次加载a.js和b.js。

// main.js
console.log('main.js ', require('./a.js').x);
console.log('main.js ', require('./b.js').x);
console.log('main.js ', require('./a.js').x);
console.log('main.js ', require('./b.js').x);

执行上面代码,结果如下。


上面代码中,第二次加载a.js和b.js时,会直接从缓存读取exports属性,所以a.js和b.js内部的console.log语句都不会执行了。

require.main

require方法有一个main属性,可以用来判断模块是直接执行还是被调用执行

直接执行的时候(node module.js),require.main属性指向模块本身。


调用执行的时候(通过require加载该脚本执行),上面的表达式返回false。

模块的加载机制

CommonJS模块的加载机制是输入的是被输出的值的拷贝。也就是说,一旦输出一个值模块内部的变化就影响不到这个值。请看下面这个例子。

下面是一个模块文件lib.js

// lib.js
var counter = 3;
function incCounter() {
counter++;
}
module.exports = {
counter: counter,
incCounter: incCounter,
};

上面代码输出内部变量counter和改写这个变量的内部方法incCounter

然后,加载上面的模块。

// main.js
var counter = require('./lib').counter;
var incCounter = require('./lib').incCounter; console.log(counter); //
incCounter();
console.log(counter); //

上面代码说明,counter输出以后,lib.js模块内部的变化就影响不到counter了。

require的内部处理流程

require命令是CommonJS规范之中,用来加载其他模块的命令。它其实不是一个全局命令,而是指向当前模块的 module.require 命令,而后者调用Node的内部命令Module._load

上面的第4步,采用module.compile()执行指定模块的脚本,逻辑如下。

上面的第1步和第2步,require函数及其辅助方法主要如下。

一旦require函数准备完毕,整个所要加载的脚本内容,就被放到一个新的函数之中,这样可以避免污染全局环境。该函数的参数包括requiremoduleexports,以及其他一些参数。

(function (exports, require, module, __filename, __dirname) {
// YOUR CODE INJECTED HERE!
});

Module._compile方法是同步执行的,所以Module._load要等它执行完成,才会向用户返回module.exports的值。

最新文章

  1. iOS-常用的第三方框架的介绍
  2. Github注册账户过程
  3. 如何把apk编译时间和最后次git commit的sha值,写入到app中
  4. Windows下Git Bash中文乱码
  5. Android 查看内存使用状况
  6. iOS UIKit:viewController之定义(2)
  7. PLSQL Developer安装(Oracle11g+win7_64bit)
  8. 【WebApi】通过HttpClient调用Web Api接口
  9. mysql把查询结果集插入到表理
  10. JDBC(14)—对DAO进行改进修改
  11. 2.基础(Foundations)
  12. 2016年蓝桥杯省赛A组c++第3题(图论)
  13. Ng第一课:引言(Introduction)
  14. bootstrap2.2相关文档
  15. 网络之Json生成解析
  16. man命令使用
  17. 设计与实现分离——面向接口编程(OO博客第三弹)
  18. java开发中beancopy比较
  19. Jquery中复选框选中取消实现文本框的显示隐藏
  20. VIN码识别,车架号识别,OCR扫描工具

热门文章

  1. 分析post与json
  2. [HEOI 2018]一双木棋
  3. win7防火墙里开启端口的图文教程 + SNMP测试感触
  4. error C2872: 'ULONG_PTR' : ambiguous symbol
  5. python 安装bs4
  6. 【CF888G】Xor-MST
  7. VMware Workstation 10 简体中文安装教程
  8. mybatis-spring多数据源配置
  9. (转) Vultr能Ping但是SSH无法连接
  10. string json list