【nodejs】exports 和 module.exports 的区别
require 用来加载代码,而 exports 和 module.exports 则用来导出代码。但很多新手可能会迷惑于 exports 和 module.exports 的区别,为了更好的理解 exports 和 module.exports 的关系,我们先来巩固下 js 的基础。示例:
app.js
var a = {name: 'nswbmw 1'};
var b = a; console.log(a);
console.log(b); b.name = 'nswbmw 2';
console.log(a);
console.log(b); var b = {name: 'nswbmw 3'};
console.log(a);
console.log(b);
运行 app.js 结果为:
{ name: 'nswbmw 1' }
{ name: 'nswbmw 1' }
{ name: 'nswbmw 2' }
{ name: 'nswbmw 2' }
{ name: 'nswbmw 2' }
{ name: 'nswbmw 3' }
解释一下:a
是一个对象,b
是对 a
的引用,即 a
和 b
指向同一个对象,即 a
和 b
指向同一块内存地址,所以前两个输出一样。当对 b
作修改时,即 a
和 b
指向同一块内存地址的内容发生了改变,所以 a
也会体现出来,所以第三、四个输出一样。当对 b
完全覆盖时,b
就指向了一块新的内存地址(并没有对原先的内存块作修改),a
还是指向原来的内存块,即 a
和 b
不再指向同一块内存,也就是说此时 a
和 b
已毫无关系,所以最后两个输出不一样。
明白了上述例子后,我们进入正题。
我们只需知道三点即可知道 exports
和 module.exports
的区别了:
exports
是指向的module.exports
的引用module.exports
初始值为一个空对象{}
,所以exports
初始值也是{}
require()
返回的是module.exports
而不是exports
所以:我们通过
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);
}
我们通常这样使用 exports
和 module.exports
一个简单的例子,计算圆的面积:
使用 exports
app.js
var circle = require('./circle');
console.log(circle.area(4));
circle.js
exports.area = function(r) {
return r * r * Math.PI;
}
使用 module.exports
app.js
var area = require('./area');
console.log(area(4));
area.js
module.exports = function(r) {
return r * r * Math.PI;
}
上面两个例子输出是一样的。你也许会问,为什么不这样写呢?
app.js
var area = require('./area');
console.log(area(4));
area.js
exports = function(r) {
return r * r * Math.PI;
}
运行上面的例子会报错。这是因为,前面的例子中通过给 exports
添加属性,只是对 exports
指向的内存做了修改,而
exports = function(r) {
return r * r * Math.PI;
}
其实是对 exports
进行了覆盖,也就是说 exports
指向了一块新的内存(内容为一个计算圆面积的函数),也就是说 exports
和 module.exports
不再指向同一块内存,也就是说此时 exports
和 module.exports
毫无联系,也就是说 module.exports
指向的那块内存并没有做任何改变,仍然为一个空对象 {}
,也就是说 area.js 导出了一个空对象,所以我们在 app.js 中调用 area(4)
会报 TypeError: object is not a function
的错误。
所以,一句话做个总结:当我们想让模块导出的是一个对象时,exports
和 module.exports
均可使用(但 exports
也不能重新覆盖为一个新的对象),而当我们想导出非对象接口时,就必须也只能覆盖 module.exports
。
我们经常看到这样的用写法:
exports = module.exports = somethings;
上面的代码等价于
module.exports = somethings;
exports = module.exports;
原因也很简单,module.exports = somethings
是对 module.exports
进行了覆盖,此时 module.exports
和 exports
的关系断裂,module.exports
指向了新的内存块,而 exports
还是指向原来的内存块,为了让 module.exports
和 exports
还是指向同一块内存或者说指向同一个 “对象”,所以我们就 exports = module.exports
。
原文地址:exports 和 module.exports 的区别。
【nodejs】exports 和 module.exports 的区别的更多相关文章
- nodejs模块中exports和module.exports的区别
通过Node.js的官方API可以看到Node.js本身提供了很多核心模块 http://nodejs.org/api/ ,这些核心模块被编译成二进制文件,可以require('模块名')去获取:核心 ...
- nodejs中exports与module.exports的区别详细介绍
如果模块是一个特定的类型就用Module.exports.如果模块是一个典型的"实例化对象"就用exports. exports.name = function() { conso ...
- nodejs里的module.exports和exports
引 在node.js中我们可以使用module.exports和exports导出模块,设置导出函数.数组.变量等等 为什么可以用这两个模块? 或者直接问,node.js的模块功能是怎么实现的. 这样 ...
- exports与module.exports的区别
nodejs有自己的模块系统,分为文件模块和内置模块.webpack是运行在node环境中,在学习vue-cli的webpack配置的时候, 发现有的文件模块: exports.fun1=functi ...
- nodeJS学习(9)--- nodeJS模块:exports vs module.exports
模块简介: 通过Node.js的官方API可以看到Node.js本身提供了很多核心模块 http://nodejs.org/api/ 这些核心模块被编译成二进制文件,可以 require('模块名') ...
- nodejs里的module.exports和exports的关系
关于node里面的module.exports和exports的异同,网上已经有很多的资料,很多的文章,很多的博客,看了很多,好像懂了,又好像不懂,过几天又不懂了...大致总结是这样的: //下面这种 ...
- node exports和module.exports区别
我们只需知道三点即可知道 exports 和 module.exports 的区别了: exports 是指向的 module.exports 的引用 module.exports 初始值为一个空对象 ...
- exports和module.exports的区别
总结:exports是module.exports的指向. 1. module应该是require方法中,上下文中的对象 2. exports对象应该是上下文中引用module.exports的新对象 ...
- nodeJS实战:自定义模块与引入,不同模块的函数传递及回调处理,exports与module.exports(基于nodejs6.2.0)
前言:本文基于上一篇文章中的源代码进行改写,地址:http://blog.csdn.net/eguid_1/article/details/52182386 注意:为什么不用module.export ...
随机推荐
- LeetCode(976. 三角形的最大周长)
问题描述 给定由一些正数(代表长度)组成的数组 A,返回由其中三个长度组成的.面积不为零的三角形的最大周长. 如果不能形成任何面积不为零的三角形,返回 0. 示例 1: 输入:[2,1,2] 输出:5 ...
- Paint the Tree
Paint the Tree 题目来源: Moscow Pre-Finals Workshop 2018 Day 5 C 题目大意: 一棵\(n(n\le2000)\)个点的树,有\(m(2<m ...
- shell脚本使用技巧2
0--stdin标准输入 1--stdout标准输出 2--stderr标准错误 重定向 echo "this is a good idea " > temp.txt tem ...
- java第十三周测试记录
今天课上遇到了问题,在我的上一篇随笔,这个阻碍了我很长时间,而且上一次也是这个问题,真的吃一堑不长一智,这次我应该就记住了,嗯. 设计思路: 俩个库: 1.一个库存商品,商品的基本属性和商品的数量(数 ...
- 【容斥+组合数】Massage @2018acm徐州邀请赛 E
问题 E: Massage 时间限制: 1 Sec 内存限制: 64 MB 题目描述 JSZKC feels so bored in the classroom that he w ...
- vue路由打开新窗口
一. <router-link>标签实现新窗口打开: 官方文档中说 v-link 指令被 <router-link> 组件指令替代,且 <router-link> ...
- adb devices 找不到设备的解决方法
1.开启adb 2.查看设备时的报错 问题1:cannot connect to daemon 解决方法:找到占据5037端口的进城,并在“任务管理器”中依据“PID”查找到,解决进程 问题2:手机 ...
- nltk 的分词器punkt: ssl问题无法下载
报错: LookupError: ********************************************************************** Resource pu ...
- Github超棒资源汇总
Awesome List 中文资源大全 经典编程书籍大全 免费的编程中文书籍索引 awesome-awesomeness-zh_CN https://github.com/jnv/lists awes ...
- Excel 驼峰表达式
=LEFT(A1,1)&MID(SUBSTITUTE(PROPER(A1),"_",""),2,100)