在node.js中我们可以使用module.exports和exports导出模块,设置导出函数、数组、变量等等

为什么可以用这两个模块?

或者直接问,node.js的模块功能是怎么实现的。

这样得益于javascript是函数性的语言,并支持闭包。

js的闭包

直接看w3cschool吧,感觉讲的挺好的:js闭包

node.js的模块实现,大致代码

首先准备一个nodejs规范的代码:

hello.js

var s = 'Hello';
var name = 'world'; console.log(s + ' ' + name + '!');

Node.js加载了hello.js后,它把这段代码包装一下,大概变成这样:

var module = {
id: 'hello',
exports: {}
};
var load = function () {
// 实际我们自己编写的hello.js代码:
function greet(name) {
console.log('Hello, ' + name + '!');
} module.exports = greet;
// hello.js代码结束
return module.exports;
};
var exports = load();
// 保存module:
save(module, exports);

module是nodejs自动加的一个对象,可见,初始化的时候会先对module.exports赋值一个空的对象{}。

save(module, exports);这个函数是个真·全局函数,作用是把exports这个变量存到某个全局变量中。其它模块通过require()函数实际上就是去这个全局变量里把对应的值拿出来。

这样,看js代码大概就明白了,为什么在nodejs里可以直接用module.exports和exports这两个语法。

module.exports和exports

module.exports和exports实际上都是对一个对象的引用,这个对象初始化就是一个空对象{}。所以直接就可以使用类似

示例一:
module.exports.foo = function () { return 'foo'; };
module.exports.bar = function () { return 'bar'; };
或者示例二:
exports.foo = function () { return 'foo'; };
exports.bar = function () { return 'bar'; };

这两个示例作用是一样的,其本质都是往最开始初始化的空数组里添加成员。

示例三
module.exports = {
hello: hello,
greet: greet
};

示例三就不一样了,实际上module.exports重新引用到了一个新的对象里。如果示例三前面有示例一或者二的代码,那么会最终导致示例一或者二导出的模块丢掉。

示例四
exports = {
hello: hello,
greet: greet
};

示例四看上去虽然和示例三差不多,但是这种写法实际上并没有输出任何变量!注意看nodejs的实现代码,load()函数里最后return的是module.exports,也就是说最后save的是module.exports的引用对象,而示例四中exports被赋值了一个新的对象,此时module.exports和exports引用的已经不是同一个对象了!

那么提问:示例三虽然对module.exports重新引用到了一个新的对象,最终结果也能实现模块的正常导出,那么示例三里的exports此时引用的是什么对象呢?

最后,如果你打算导出一个数组或者变量,或者函数,都会涉及到module.exports原引用对象的丢弃,要额外注意,此时要小心不要丢掉前面已经导出的模块。

结论

如果要输出一个键值对象{},可以利用exports这个已存在的空对象{},并继续在上面添加新的键值;

如果要输出一个函数或数组,必须直接对module.exports对象赋值。

所以我们可以得出结论:直接对module.exports赋值,可以应对任何情况:

module.exports = {
foo: function () { return 'foo'; }
};
或者:
module.exports = function () { return 'foo'; };

最终,我们强烈建议使用module.exports = xxx的方式来输出模块变量,这样,你只需要记忆一种方法。

或者使用我喜欢的方法,对空对象直接添加值:

var foo = function () { return 'foo'; };

...

module.exports.foo = foo;//
module.exports.bar = function () { return 'bar'; };

参考

liaoxuefeng

exports 和 module.exports 的区别

w3cschool

JS是按值传递还是按引用传递?,但是我不是很同意文中所说的按共享传递这名字的叫法。实际上无论传递普通变量还是传递函数,都是按值拷贝传递,只不过传递对象的时候,拷贝过去的是个引用变量罢了,即引用副本。

这里就是c语言指针的思想。

如果你对我上面所讲的还不明白,建议去把c语言里的指针好好的重学一遍。真正理解透彻了c里面的指针,学起来其它所有语言都不怕。

nodejs里的module.exports和exports的更多相关文章

  1. nodejs里的module.exports和exports的关系

    关于node里面的module.exports和exports的异同,网上已经有很多的资料,很多的文章,很多的博客,看了很多,好像懂了,又好像不懂,过几天又不懂了...大致总结是这样的: //下面这种 ...

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

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

  3. 关于common.js里面的module.exports与es6的export default的思考总结

    背景 公司项目需要裁切功能,基于第三方图片裁切组件vue-cropper(0.4.0版本),封装了图片裁切组件(picture-cut)(放在公司内部组件库,仅限于公司内部使用) 在vue-cropp ...

  4. module.exports 与 exports

    module.exports 与 exports 注意:1 对于要导出的属性,可以简单直接挂到 exports 对象上2 对于类,为了直接使导出的内容作为类的构造器可以让调用者使用 new 操作符创建 ...

  5. module.exports和exports得区别

    对module.exports和exports的一些理解 可能是有史以来最简单通俗易懂的有关Module.exports和exports区别的文章了. exports = module.exports ...

  6. (转)Node.js module.exports与exports

    本文转自Node.js module.exports与exports 作者: chemdemo 折腾Node.js有些日子了,下面将陆陆续续记录下使用Node.js的一些细节. 熟悉Node.js的童 ...

  7. module.exports和exports

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

  8. Module.exports和exports的区别

    原文链接: https://www.ycjcl.cc/2017/02/10/module-exportshe-exportsde-qu-bie/ 学习Seajs时,看到了exports.doSomet ...

  9. module.exports,exports和export default,export的区别

    前提:CommonJS模块规范和ES6模块规范是完全不同的两个概念. module.exports,exports属于CommonJS模块规范: export default,export属于ES6模 ...

随机推荐

  1. 201521123006 《Java程序设计》 第2周学习总结

    1. 本周学习总结 本周进一步学习了java,了解了java编程中一些特定的用法,比如:在编译程序时可以使用import来减少输入包名称.本周还学会了使用枚举类(enum Choice{fab,sor ...

  2. 201521123069 《Java程序设计》 第14周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多数据库相关内容. 2. 书面作业 1. MySQL数据库基本操作 建立数据库,将自己的姓名.学号作为一条记录插入.(截图,需出现自 ...

  3. 学习Python不得不关注和学习的国外大神博客

    注意 : 本文收集于网路 . 由于常常更新 , 有些链接打不开, 请自备梯子 在学习Python过程中,总会遇到各种各样的坑, 虽然Python是一门优美而简单易学的语言 . 但当学习后 , 总想着更 ...

  4. Hibernate中的主键生成器generator

    本文讲述Hibernate的generator属性的意义.Generator属性有7种class,本文简略描述了这7种class的意义和用法. [xhtml] view plaincopy <c ...

  5. python3中的编码与解码用法

    #!/usr/bin/env python3 # -*- coding: utf-8 -*- __author__ = '人生入戏' #python3在编码时会把str编码成utf-8的bytes类型 ...

  6. “AOP代理”遇到“双上下文”

    最近有小伙伴儿遇到了一个问题来咨询我,问题大致如下: 他在Service层利用Aspect设置了一个Spring AOP代理,在单元测试以及在service层代码上添加代理的时候均没有发现问题,但是在 ...

  7. Java-Filter过滤器用于过滤整个项目的编码

    整个分为实现类以及在web.xml文件中对编写的filter类进行注册 代码如下 package cn.itcast.itcaststore.web.filter; import java.io.IO ...

  8. 关于Visio Studio 2012使用Nuget获取Sqlite驱动包报错:“System.Data.SQLite.EF6”的架构版本与 NuGet 的版本 2.0.30625.9003 不兼容

    背景 笔者的VS2012版本比较老旧,是几年以前下载的.平时添加三方包和驱动包都是手动添加.后来了解到有Nuget这个工具,如获至宝.可是在使用过程中却出了不少问题. 最初,笔者尝试使用Nuget添加 ...

  9. 如何保存或读取数据(到android的data目录)利用context获取常见目录可优化代码

    读取用户信息 当然这里可以有多种返回值 非硬性

  10. sdk&jdk&jre

    1. jre and jdkJRE(Java Runtime Enviroment)是Java的运行环境.面向Java程序的使用者,而不是开发者.如果你仅下载并安装了JRE,那么你的系统只能运行Jav ...