[摘抄] 4.require命令

4.require命令

1. 基本用法

Node适用CommonJS模块规范,内置的require命令用于加载模块文件。javascript

require命令的基本功能是,读入并执行一个JavaScript文件,而后返回该模块的exports对象。 若是没有发现指定模块,就会报错。java

var invisible = function(){
    console.log('invisible');
}

exports.message = 'hi';

exports.say = function (){
    console.log(message);
}

运行下面的命令,能够输出exports对象。node

var example = require('./example.js');

example{
    message:'hi',
    say:[Function]
}

若是模块输入的是一个函数,那就不能定义在exports对象上面,而要定义在module.exports变量上面。json

module.exports = function (){
  console.log('hello world')
}
require('./example.js')()

上面代码中,require命令调用自身,等因而执行module.exports,所以会输出"hello world"。缓存

4.2 加载规则

require命令用于加载文件,后缀名默认为.jsbash

var foo = require('foo');
//  等同于
var foo = require('foo.js');

根据参数的不一样格式,require命令去不一样路径寻找模块文件。函数

(1)若是参数字符串以“/”开头,则表示加载的是一个位于绝对路径的模块文件。好比,require('/home/marco/foo.js')将加载/home/marco/foo.jsui

(2)若是参数字符串以“./”开头,则表示加载的是一个位于相对路径(跟当前执行脚本的位置相比)的模块文件。好比,require('./circle')将加载当前脚本同一目录的circle.js设计

(3)若是参数字符串不以“./“或”/“开头,则表示加载的是一个默认提供的核心模块(位于Node的系统安装目录中),或者一个位于各级node_modules目录的已安装模块(全局安装或局部安装)。code

举例来讲,脚本/home/user/projects/foo.js执行了require('bar.js')命令,Node会依次搜索如下文件。

  • /usr/local/lib/node/bar.js
  • /home/user/projects/node_modules/bar.js
  • /home/user/node_modules/bar.js
  • /home/node_modules/bar.js
  • /node_modules/bar.js

这样设计的目的是,使得不一样的模块能够将所依赖的模块本地化。

(4)若是参数字符串不以“./“或”/“开头,并且是一个路径,好比require('example-module/path/to/file'),则将先找到example-module的位置,而后再以它为参数,找到后续路径。

(5)若是指定的模块文件没有发现,Node会尝试为文件名添加.js.json.node后,再去搜索。.js件会以文本格式的JavaScript脚本文件解析,.json文件会以JSON格式的文本文件解析,.node文件会以编译后的二进制文件解析。

(6)若是想获得require命令加载的确切文件名,使用require.resolve()方法。

4.3 目录的加载机制

一般,咱们会把相关的文件会放在一个目录里面,便于组织。这时,最好为该目录设置一个入口文件,让require方法能够经过这个入口文件,加载整个目录。

在目录中放置一个package.json文件,而且将入口文件写入main字段。下面是一个例子。

// package.json
{ "name" : "some-library",
  "main" : "./lib/some-library.js" }

require发现参数字符串指向一个目录之后,会自动查看该目录的package.json文件,而后加载main字段指定的入口文件。若是package.json文件没有main字段,或者根本就没有package.json文件,则会加载该目录下的index.js文件或index.node文件。

4.4 模块的缓存

第一次加载某个模块时,Node会缓存该模块。之后再加载该模块时就直接从缓存取出该模块的module.exports属性。

require('./example.js');
require('./example.js').mesage = 'hello';
require('./example.js').message;
// "hello"
  • 上面代码中,连续三次使用require命令,加载同一个模块。
  • 第二次加载的时候,为输出的对象添加了一个message属性。
  • 第三次加载的时候,这个message属性依然存在,这就证实require命令并无被从新加载,而是输出了缓存。

若是想要屡次执行某个模块,可让该模块输出一个函数,而后每次require这个模块的时候,从新执行一下输出函数。

全部缓存的模块保存在require.cache之中,若是想删除模块的缓存,能够像下面这样写。

// 删除指定的模块缓存
delete require.cache[moduleName];
// 删除全部模块的缓存
Objcet.keys(require.cache).forEach(function(key){
    delete require.cache[key];
})

注意,缓存是根据绝对路径识别模块的,若是同的模块名,可是保存在不一样的路径,require命令仍是会从新加载该模块。

4.5 环境变量NODE_PATH

Node执行一个脚本时,会先查看环境变量NODE_PATH。他是一组以冒号分隔的绝对路径。在其余位置找不到指定模块时,Node会去这些路径查找。

能够将NODE_PATH添加到.bashrc

export NODE_PATH="/usr/local/lib/node"

因此,若是遇到复杂的相对路径,好比下面这样

var myModule = require('../../../../lib/myModule');

有两种解决方法,

  • 一是将该文件加入node_modules 目录
  • 二是修改NODE_PATH环境变量

package.json文件能够采用下面的写法

{
  "name": "node_path",
  "version": "1.0.0",
  "description": "",
  "main": "index.js",
  "scripts": {
    "start": "NODE_PATH=lib node index.js"
  },
  "author": "",
  "license": "ISC"
}

NODE_PATH是历史遗留下来的一个路径解决方案,一般不该该使用,而应该使用node_modules目录机制。

4.6 模块的循环加载

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

// a.js
// 【2】.文件a在exports中建立变量并赋值为`a1`
exports.x = 'a1';
// 【3】.导入文件b,文件b开始执行。
console.log('a.js ', require('./b.js').x); //b2
// 【7】.b文件执行完毕,a文件继续往下执行,赋值,require.cache中a文件的x值变为a2;
exports.x = 'a2';

// b.js
// 【4】.执行文件b,建立变量并赋值`b1`
exports.x = 'b1';
// 【5】.导入文件a,文件a已经被执行过,因此在[require.cache]中是有a文件的缓存,而且exports.x = a1,下面则不会再执行a文件而是从缓存中获得x值,为a1;
console.log('b.js ', require('./a.js').x); //a1
// 【6】.赋值,执行完毕
exports.x = 'b2';

// main.js
// 【1】.开始读取文件a
console.log('main.js ', require('./a.js').x); //a2
// 【8】.a文件读取完毕,往下执行读取b文件,b文件在a文件的执行过程当中已经读取,则拿出缓存直接打印 
console.log('main.js ', require('./b.js').x); //b2

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

$ node main.js //开始执行
b.js  a1
a.js  b2
main.js  a2
main.js  b2

修改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);

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

$ node main.js
b.js  a1
a.js  b2
main.js  a2
main.js  b2
main.js  a2
main.js  b2

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

4.7 require.main

require方法有一个main属性,能够用来判断模块是直接执行,仍是被调用执行。

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

require.main === module
// true

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

相关文章
相关标签/搜索