commonJS

hello world2

commonJS是一个规范

commonJS是一个规范,讲起来内容会有非常多,我们来点直接的要点

CommonJS对模块的定义十分简单,主要分为模块引用、模块定义和模块标识3个部分。

核心如下

  • require - 用来引入依赖
  • export - 用来导出模块,包括标识符(identifier)和模块内容(contents)
    • module.exports
    • exports.xxx

下面以helloworld2来演示具体的用法

关于 helloworld2

helloworld2会有2个文件,用来演示多文件直接引用关系

  • helloworld2.js
  • main.js

创建helloworld2.js

  1. module.exports = function () {
  2. console.log('hello world');
  3. }

这里使用module.exports导出一个function,里面只是打印了hello world

下面我们看一下如何在main.js里调用

创建main.js

  1. var hello = require('./helloworld2');
  2. hello()

这样需要说明的是,通过require来引用helloworld2这个模块,一旦你require了这个模块,那么这个模块对外暴露的方法或变量,你就可以调用了

再回想一下module.exports的作用:当前module暴露的方法或变量

那么此处我们可能会有的疑问

  • require引用的是一个本地文件(注意./helloworld2代表的是当前目录下的helloworld2.js),如果是其他库呢?
  • module.exports这里只是暴露了一个function,那如果想暴露更多方法或变量呢?

执行main.js

  1. nodejs git:(master) node demo/main.js
  2. hello world

我们如果再变一下呢?

helloworld2这个module导出的是一个function,既然是function那么就一定可以直接调用

创建main.js

  1. require('./helloworld2')();

执行

  1. nodejs git:(master) node demo/main2.js
  2. hello world

那么我如果想跟某人打招呼呢?

创建helloworld3.js

  1. module.exports = function (person) {
  2. console.log('hello world ' + person);
  3. }

创建main3.js

  1. require('./helloworld3')('海角');

执行

  1. nodejs git:(master) node demo/main3.js
  2. hello world 海角

我想调用独立函数

创建helloworld4.js

  1. function say(person) {
  2. console.log('hello world ' + person);
  3. }
  4. module.exports = say;

创建main4.js

  1. require('./helloworld3')('海角');

执行

  1. nodejs git:(master) node demo/main4.js
  2. hello world 海角

解答疑问

require引用的是一个本地文件,如果是其他库呢?

(注意./helloworld2代表的是当前目录下的helloworld2.js)

module.exports这里只是暴露了一个function,那如果想暴露更多方法或变量呢?

我们来假设一下,在helloworld5里既能吃饭又能打招呼呢?

也就是说这个模块要提供2个方法

  • 吃饭
  • 打招呼

创建helloworld5.js

  1. function say(person) {
  2. console.log('i am say hello world to ' + person);
  3. }
  4. function eat(food) {
  5. console.log('i am eat ' + food);
  6. }
  7. exports.eat = eat;
  8. exports.say = say;

创建main5.js

  1. var h5 = require('./helloworld5')
  2. h5.eat('兰州拉面');
  3. h5.say('海角');

执行

  1. nodejs git:(master) node demo/main5.js
  2. i am eat 兰州拉面
  3. i am say hello world to 海角

如果你还记得之前的模块是如何导出,你就会对比一下

之前用的是module.exports,现在用的是exports.xxx

那么你好奇,它们有什么差别么?

module-exports

比较一下module.exports 和exports.xxx

module.exports才是真正的接口,exports只不过是它的一个辅助工具。 最终返回给调用的是module.exports而不是exports。

所有的exports收集到的属性和方法,都赋值给了module.exports。

当然,这有个前提,就是Module.exports本身不具备任何属性和方法。如果module.exports已经具备一些属性和方法,那么exports收集来的信息将被忽略

共存的噩梦

修改exports.js如下:

  1. module.exports = 'Exports IT!';
  2. exports.name = function () {
  3. console.log('hello world');
  4. }

再次引用执行e_main.js

  1. var hello = require('./exports');
  2. console.log(hello.name())

发现报错:对象hello没有name方法

exports模块忽略了exports收集的name方法,返回了一个字符串“Exports IT!”。

由此可知,你的模块并不一定非得返回“实例化对象”。

你的模块可以是任何合法的javascript对象—boolean, number, date, JSON, string, function, array等等。

你的模块可以是任何你设置给它的东西。如果你没有显式的给module.exports设置任何属性和方法,那么你的模块就是exports设置给module.exports的属性。

string

  1. exports.name = "StuQ微课堂"

boolean

布尔值就是真或者假,在js表示为true或false

  1. exports.is_valid = true

number

数值有很多种,大家记得它是描述数字相关的即可

常见如整形,浮点型

年龄

  1. exports.age = 20

圆周率

  1. var pi = 3.1415;
  2. exports.PI = pi

date

日期类型

  1. var date = new Date()
  2. exports.created_at = date;

那么怎么样获取2015-11-20呢?

使用momentjs

首先需要安装momentjs

  1. npm install --save moment

date.js实例代码

  1. var moment = require('moment');
  2. var date = moment().format('YYYY-MM-DD');
  3. exports.created_at = date;

测试main.js

  1. var date = require('./date');
  2. console.log(date.created_at);

执行结果如下

  1. nodejs-practice git:(master) node nodejs/demo/date/main.js
  2. 2015-11-20

JSON

json.js

  1. exports.person = {
  2. name: "西瓜",
  3. age:18
  4. }

创建main.js

  1. var json = require('./json')
  2. console.log(json.person.name);
  3. // console.log(json.persion);
  4. for(var k in json.person){
  5. console.log("key = " + k + " && value = " + json.person[k]);
  6. }

最简单的面向对象

下面例子中,你的模块是一个类exports2.js:

  1. module.exports = function(name, age) {
  2. this.name = name;
  3. this.age = age;
  4. this.about = function() {
  5. console.log(this.name +' is '+ this.age +' years old');
  6. };
  7. };

可以这样应用它:

  1. var Person = require('./exports2');
  2. var p = new Person('Ting', 20);
  3. p.about(); // Ting is 20 years old

数组

下面例子中,你的模块是一个数组:

  1. module.exports = ['eric', '海角', '西瓜', '李小贱', 'kevin'];

可以这样应用它:

  1. var arr = require('./exports3.js');
  2. console.log('第3个是西瓜: ' + arr[2]);
  3. console.log('-------------------\n');
  4. for(var i = 0; i < arr.length; i++){
  5. console.log("第" + i + "个是 " + arr[i]);
  6. }
  7. console.log('-------------------\n');
  8. for(var i = 1; i < arr.length; i++){
  9. console.log("第" + i + "个是 " + arr[i]);
  10. }
  11. console.log('-------------------\n');
  12. var i = 0;
  13. arr.forEach(function(name){
  14. console.log("第" + i + "个是 " + name);
  15. i++;
  16. })

现在你应该明白了,如果你想你的模块是一个特定的类型就用module.exports。

如果你想的模块是一个典型的“实例化对象”就用exports。

给Module.exports添加属性类似于给exports添加属性。例如:

  1. module.exports.name = function() {
  2. console.log('My name is Lemmy Kilmister');
  3. };

同样,exports是这样的

  1. exports.name = function() {
  2. console.log('My name is Lemmy Kilmister');
  3. };

请注意,这两种结果并不相同。

前面已经提到module.exports是真正的接口,exports只不过是它的辅助工具。

推荐使用exports导出,除非你打算从原来的“实例化对象”改变成一个类型

dependency cycle循环引用

a.js

  1. console.log('a starting');
  2. exports.done = false;
  3. var b = require('./b.js');
  4. console.log('in a, b.done = %j', b.done);
  5. exports.done = true;
  6. console.log('a done');

b.js

  1. console.log('b starting');
  2. exports.done = false;
  3. var a = require('./a.js');
  4. console.log('in b, a.done = %j', a.done);
  5. exports.done = true;
  6. console.log('b done');

main.js

  1. console.log('main starting');
  2. var a = require('./a.js');
  3. var b = require('./b.js');
  4. console.log('in main, a.done=%j, b.done=%j', a.done, b.done);

当main.js加载a.js的时候,a.js加载b.js,同时,b.js想要加载a.js,这时候就产生了依赖闭环的问题,为了避免无限循环,需要打破这个闭环。根据CommonJS Modules/1.0规范中的说明「在这种情况下,”require”返回的对象必须至少包含此外部模块在调用require函数(会进入当前模块执行环境)之前就已经准备完毕的输出。」,有些绕,让我们从依赖闭环产生的地方跟踪,b.js需要require a.js,这里b.js做为当前模块,a.js相对于b.js来说是外部模块,那么a.js的输出应该是在其require b.js之前(即「进入当前模块执行环境」)就应该返回,执行过程如下:

a.js

  1. console.log('a starting');
  2. exports.done = false;
  3. // 只执行到这里,然后exports返回给调用模块(b.js),以下被丢弃
  4. var b = require('./b.js');
  5. console.log('in a, b.done = %j', b.done);
  6. exports.done = true;
  7. console.log('a done');

然后b.js继续执行完成。以下是执行结果:

  1. $ node main.js
  2. main starting
  3. a starting
  4. b starting
  5. in b, a.done = false
  6. b done
  7. in a, b.done = true
  8. a done
  9. in main, a.done=true, b.done=true

注意,虽然main.js同时require了a.js和b.js,但是根据node.js的模块缓存策略,模块只执行一次。