模块简介:

通过Node.js的官方API可以看到Node.js本身提供了很多核心模块 http://nodejs.org/api/

  • 这些核心模块被编译成二进制文件,可以 require('模块名') 去获取;
  • 核心模块具有最高的加载优先级(有模块与核心模块同名时会体现)

(本次主要说自定义模块)

Node.js还有一类模块为文件模块,可以是 JavaScript 代码文件(.js作为文件后缀)、也可以是 JSON 格式文本文件(.json作为文件后缀)、还可以是编辑过的 C/C++ 文件(.node作为文件后缀);

文件模块访问方式通过require('/文件名.后缀')   ; require('./文件名.后缀')   ; requrie('../文件名.后缀') 去访问,文件后缀可以省略;以"/"开头是以绝对路径去加载,以"./"开头和以"../"开头表示以相对路径加载,而以"./"开头表示同级目录下文件,

前面提到文件后缀可以省略,Nodejs 尝试加载的优先级 js文件 > json文件 > node文件

创建一个自定义模块

以一个计数器为例

var outputVal  = ;     //输出值
var increment = ; //增量 /* 设置输出值 */
function seOutputVal (val) {
outputVal = val;
} /* 设置增量 */
function setIncrement(incrementVal){
increment = incrementVal;
} /* 输出 */
function printNextCount()
{
outputVal += increment;
console.log(outputVal) ;
} function printOutputVal() {
console.log(outputVal);
}
exports.seOutputVal = seOutputVal;
exports.setIncrement = setIncrement; module.exports.printNextCount = printNextCount;

自定义模块 示例源码

示例中重点在于exports和module.exports;提供了外部访问的接口,下面调用一下看看效果吧

调用自定义模块

/*
一个Node.js文件就是一个模块,这个文件可能是Javascript代码、JSON或者编译过的C/C++扩展。
重要的两个对象:
require是从外部获取模块
exports是把模块接口公开
*/
var counter = require('./1_modules_custom_counter'); console.log('第一次调用模块[1_modules_custom_counter]'); counter.seOutputVal(); //设置从10开始计数
counter.setIncrement (); //设置增量为10 counter.printNextCount();
counter.printNextCount();
counter.printNextCount();
counter.printNextCount(); /*
require多次调用同一模块不会重复加载
*/
var counter = require('./1_modules_custom_counter'); console.log('第二次调用模块[1_modules_custom_counter]');
counter.printNextCount();

自定义模式调用 源码

运行可以发现通过 exports 和 module.exports 对外公开的方法都可以访问!

示例中可以看到,我两次通过require('./1_modules_custom_counter')获取模块,但是第二次引用后调用 printNextCount()方法确从60开始~~~

原因是node.js通过require,require多次调用同一模块不会重复加载,Node.js会根据文件名缓存所有加载过的文件模块,所以不会重新加载了

注意:通过文件名缓存是指实际文件名,并不会因为传入的路径形式不一样而认会是不同的文件

在我创建的1_modules_custom_counter文件中有一个printOutputVal()方法,它并没有通过exports或module.exports提供对外公开访问方法,

如果1_modules_load文件中直接访问运行会出现什么样的情况呢?

答案是:TypeError: Object #<Object> has no method 'printOutputVal'

require, exports, module.exports 区别:

  • require 用来加载代码(模块),而 exports 和 module.exports 则用来导出代码。

    exports 是指向的 module.exports 的引用
    module.exports 初始值为一个空对象 {},所以 exports 初始值也是 {}
    require() 返回的是 module.exports 而不是 exports

nodejs中exports与module.exports的区别:

API提供了解释 http://nodejs.org/api/modules.html

Note that exports is a reference to module.exports making it suitable for augmentation only. If you are exporting a single item such as a constructor you will want to use module.exports directly instead
exports 仅仅是 module.exports 的一个地址引用。nodejs 只会导出 module.exports 的指向,如果 exports 指向变了,那就仅仅是 exports 不在指向 module.exports,于是不会再被导出
  • Module.exports:如果你想你的模块是一个 特定的类型 就用 Module.exports。
  • exports: 如果你想的模块是一个 典型的“实例化对象” 就用 exports。

原文:http://www.hacksparrow.com/node-js-exports-vs-module-exports.html

参考:http://www.cnblogs.com/kongxianghai/p/3936197.html

http://www.cnblogs.com/zhongweiv/p/nodejs_module.html

1. nodejs 模块中的 exports 对象:你可以用它创建你的模块。例如:

假设这是 rocker.js 文件:

# rocker.js 
exports.name = function() {
console.log('My name is Lemmy Kilmister');
};

则,在另一文件中,你可以这样引用:

var rocker = require('./rocker.js');
rocker.name(); // 'My name is Lemmy Kilmister'

强调:

(1)给 exports 赋值其实是给 module.exports 这个空对象添加了属性,eg:


var name = 'nswbmw';
exports.name = name;
exports.sayName = function() { console.log(name); } # 给 exports 赋值其实是给 module.exports 这个空对象添加了两个属性而已,上面的代码相当于:
var name = 'nswbmw';
module.exports.name = name;
module.exports.sayName = function() { console.log(name); }

(2)两者的使用方法不同,eg:

使用 exports:

# circle.js -- 使用 exports 定义一个方法
exports.area = function(r) { return r * r * Math.PI; }
# app.js -- 使用上面文件里定义的方法
var circle = require('./circle');
console.log(circle.area(4)); # 使用,引用的方式:***.area()

使用 Module.exports:

# area.js -- 使用 module.exports 定义方法
module.exports = function(r) { return r * r * Math.PI;
# app.js -- 使用上面文件定义的方法
var area = require('./area');
console.log(area(4)); # 直接使用,***()

2. Module.exports :

  • Module.exports 才是真正的接口,exports只不过是它的一个辅助工具。
  • 最终返回给调用的是Module.exports而不是exports。
  • 所有的exports收集到的属性和方法,都赋值给了Module.exports。当然,这有个前提,就是Module.exports本身不具备任何属性和方法。如果,Module.exports已经具备一些属性和方法,那么exports收集来的信息将被忽略。eg:修改 rocker.js 如下:
    # 修改的 rocker.js
    module.exports = 'ROCK IT!'; # 将 module.exports = {}; 变为了 module.exports = '';
    exports.name = function() {
    console.log('My name is Lemmy Kilmister');
    };

    此时,再次引用执行 rocker.js 如下:

    var rocker = require('./rocker.js');
    rocker.name(); // TypeError: Object ROCK IT! has no method 'name'
    # 发现报错:对象“ROCK IT!”没有name方法 

    rocker 模块忽略了 exports 收集的 name 方法,返回了一个字符串“ROCK IT!”。由此可知,你的模块并不一定非得返回“实例化对象”。你的模块可以是任何合法的javascript对象--boolean, number, date, JSON, string, function, array等等。

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

  eg1: 你的模块是一个类:

# 再次修改 rocker.js
module.exports = function(name, age) {
this.name = name;
this.age = age;
this.about = function() {
console.log(this.name +' is '+ this.age +' years old');
};
};

可以这样使用它:

var Rocker = require('./rocker.js');
var r = new Rocker('Ozzy', 62);
r.about(); // Ozzy is 62 years old

eg2: 你的模块是一个数组:

# 再次修改 rocker.js
module.exports = ['Lemmy Kilmister', 'Ozzy Osbourne', 'Ronnie James Dio', 'Steven Tyler', 'Mick Jagger'];

可以这样使用它:

var rocker = require('./rocker.js');
console.log('Rockin in heaven: ' + rocker[2]); //Rockin in heaven: Ronnie James Dio

现在你明白了,如果你想你的模块是一个特定的类型就用 Module.exports,直接 变量/new方法 使用。如果你想的模块是一个典型的“实例化对象”就用 exports,需引用使用 .***

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

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

    同样, exports 是这样的:

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

    强调:这两种定义效果并不相同(本人还是有点晕,后续将继续学习)。前面已经提到 module.exports 是真正的接口,exports 只不过是它的辅助工具。推荐使用 exports 导出,除非你打算从原来的 “实例化对象” 改变成一个 类型!!

exports和module.exports 覆盖

上面也也基本明白了exports和module.exports的关系和区别,但如果同时针对printNextCount()方法存在exports和module.exports,结果如何?

调用结果

从结果可以看出,并没有报错,表示可以这么定义,但最终 module.exports 覆盖了exports

****虽然结果不会报错,如果这么用开发中难免会有一些问题存在,所以****

1. 最好别分别定义 module.exports 和 exports

2. NodeJs 开发者建议导出对象用 module.exports, 导出多个方法和变量用 exports

其它...

  API中还提供了其它的方法,就不细讲了,在上面例子的基础上自已动手一输出就知道了

module.id:返回string类型的模块标识,一般为完全解析后的文件名

module.filename:返回一个string类型的完全解析后文件名

module.loaded:返回一个bool类型,表示是否加载完成

module.parent:返回引用该模块的模块

module.children:返回该模块引用的所有模块对象的数组

nodeJS学习(9)--- nodeJS模块:exports vs module.exports的更多相关文章

  1. nodejs模块中exports和module.exports的区别

    通过Node.js的官方API可以看到Node.js本身提供了很多核心模块 http://nodejs.org/api/ ,这些核心模块被编译成二进制文件,可以require('模块名')去获取:核心 ...

  2. 通过Anuglar Material串串学客户端开发 - NodeJS模块机制之Module.Exports

    module.exports 前文讲到在Angular Material的第二个编译文件docs/gulpfile.js中却看到了一个奇怪的东西module.exports那么module.expor ...

  3. nodeJS实战:自定义模块与引入,不同模块的函数传递及回调处理,exports与module.exports(基于nodejs6.2.0)

    前言:本文基于上一篇文章中的源代码进行改写,地址:http://blog.csdn.net/eguid_1/article/details/52182386 注意:为什么不用module.export ...

  4. node (02 CommonJs 和 Nodejs 中自定义模块)顺便讲讲module.exports和exports的区别 dependencies 与 devDependencies 之间的区别

    CommonJS 规范的提出,主要是为了弥补当前 JavaScript 没有标准的缺陷.它的终极目标就是:提供一个类似 Python,Ruby 和 Java 语言的标准库,而不只是停留在小脚本程序的阶 ...

  5. NodeJS的exports、module.exports与ES6的export、export default深入详解

    前言 决定开始重新规范的学习一下node编程.但是引入模块我看到用 require的方式,再联想到咱们的ES6各种export .export default. 阿西吧,头都大了.... 头大完了,那 ...

  6. nodejs中exports与module.exports的区别详细介绍

    如果模块是一个特定的类型就用Module.exports.如果模块是一个典型的"实例化对象"就用exports. exports.name = function() { conso ...

  7. 【nodejs】exports 和 module.exports 的区别

    require 用来加载代码,而 exports 和 module.exports 则用来导出代码.但很多新手可能会迷惑于 exports 和 module.exports 的区别,为了更好的理解 e ...

  8. nodejs中的exports和module.exports

    module是随文件而生的全局变量,它有exports属性,这个属性默认是一个空的字典. node的设计者画蛇添足有定义了一个exports全局变量,export指向module.exports所指向 ...

  9. Node.js学习之(第二章:exports和module.exports)

    前言 Node中,每个模块都有一个exports接口对象,我们需要把公共的方法或者字符串挂载在这个接口对象中,其他的模块才可以使用. Node.js中只有模块作用域,默认两个模块之间的变量,方法互不冲 ...

随机推荐

  1. 安装配置eclipse的图文步骤

    装eclipse 之前要确定自己是否已经安装了java开发环境JDK,JDK的版本64位要下载Eclipse版本64位:JDK32位,要下载Eclipse32位. 一.去eclipse官网下载ecli ...

  2. SQL初次接触

    1.SQL对大小写不敏感 2.部分SQL数据库要求结尾分号 3.分为两种DML(数据操作语言)和DDL(数据定义语言) sql中一些注意要点 1.设置主键 一般会在一个数据内设置一个主键(名字通常为i ...

  3. linux关于权限

    用户权限:drwxr-x---. 8 root root 4096 8月 6 23:18 mnt 第一个root:所有者 即root用户第二个root:所有者所在的组mnt:所有者创建的文件夹Rwx: ...

  4. Linux 下上传下载命令,SCP,SFTP,FTP

    scp 帮助命令: man scp scp功能: 下载远程文件或者目录到本地, 如果想上传或者想下载目录,最好的办法是采用tar压缩一下,是最明智的选择. 从远程主机 下载东西到 本地电脑 拷贝文件命 ...

  5. React学习记录一

    半路出家直接上手React,其实有点吃力,所以开始研究create-react-app,从这里下手吧. create-react-app 官方网站:https://github.com/faceboo ...

  6. Node 操作MySql数据库

    1, 需要安装 MySQL 依赖 =>  npm i mysql -D 2, 封装一个工具类 mysql-util.js // 引入 mysql 数据库连接依赖 const mysql = re ...

  7. [转载]本地配置的 *.dev,*.app域名 在谷歌浏览器中总是自动转跳到https上,导致不能访问?

    本地开发环境 .dev 不正常,找到文章mark一下 转自:https://segmentfault.com/q/1010000012339191

  8. 微信小程序中 this.setData is not a function报错

    在微信小程序中我们一般通过以下方式来修改data中的数据: 比如获取小程序缓存: wx.getStorage({ key: 'is_screen', success: function (res) { ...

  9. Python知识点入门笔记——特色数据类型(字典)

    Python中字典是一些列键 (key) 值 (value) 对组成的,形式如下 {键1:值1,键2:值:2} 每个键值对用冒号隔开,每对之间用逗号隔开 键必须是唯一的,必须是不可变的,如字符串,数字 ...

  10. mysql同步故障解决

    故障现象:Slave_SQL_Running: No Slave状态:mysql> show slave status\GSlave_IO_Running: YesSlave_SQL_Runni ...