目录
3.向外共享模块作用域中的成员(exports和module.exports)
一、什么是模块化
编程领域中的模块化,就是遵守固定的规则,把一个大文件拆成独立并相互依赖的多个小模块
模块化的有点:
①提高了代码的复用性
②提高了代码的可维护性
③可以实现按需加载
二、模块化规范
模块化规范就是对代码进行模块化的拆分与组合时,需要遵守的那些规则
例如;
①使用什么样的语法格式来引用模块
②在模块中使用什么样的语法格式向外暴露成员
模块化规范的好处:大家都遵守同样的模块化规范写代码,降低了沟通的成本,极大方便了各个模块之间的相互调用,利人利己
三、Node.js中模块的分类
Node.js中根据模块来源的不同,讲模块分为了3大类,分别是:
①内置模块(内置模块是有Node.js官方提供的,例如fs、path、http等)
②自定义模块(用户创建的每个.js文件,都是自定义模块)
③第三方模块(由第三方开发出来的模块,并非官方提供的内置模块,也不是用户创建的自定义模块,使用前需要现在)
四、Node.js中的模块作用域
1.什么是模块作用域
和函数作用域类似,在定义模块中定义的变量,方法等成员,只能在当前模块内被访问,这种模块级别的访问限制,叫做模块作用域
2.模块作用域的好处
防止了全局变量污染的问题
3.向外共享模块作用域中的成员(exports和module.exports)
①module对象
在每个.js自定义模块中都有一个module对象,它里面存储了和当前模块有关的信息,打印如下:
②module.exports对象
在自定义模块中,可以使用module.exports对象,将模块内的成员共享出去,供外界使用。外界用require()方法导入自定义模块时,得到的就是module.export所指向的对象。
③共享成员的注意点
使用require()方法导入模块时,导入的结果,永远以module.exports指向的对象为准
④exports对象
由于module.exports单词写起来比较复杂,为了简化向外共享成员的代码,Node提供了exports对象,默认情况下,exports和module.exports指向同一个对象,最终共享的结果,还是以module.exports指向的对象为准
⑤exports和module.exports的使用误区
使用require()模块时,得到的永远是module.exports指向的对象:
注意:为了防止混乱,建议不要在同一个模块中同时使用exports和module.exports
⑥Node.js中的模块化规范
Node.js遵循了CommonJs模块化规范,CommonJS规定了模块的特性和各模块之间如何相互依赖.
CommonJs规定:
1.每个模块内部,module变量代表当前模块
2.module变量是一个对象,它的exports属性(及module.exports)是对外的接口
3.加载某个模块时,其实是加载该模块的module.exports属性,require()方法用于加载模块
五、模块的加载机制
1.如何加载模块
使用强大的require(),可以加载需要的内置模块,用户自定义模块,第三方模块进行使用
示例:
注意:使用require()方法加载其它模块时,会执行被加载模块中的代码
2.优先从缓存中加载
模块在第一次加载后会被缓存,这也意味着多次调用require()不会导致模块的代码被执行多次.
注意:不论是内置模块,用户自定义模块、还是第三方模块,他们都会优先从缓存中加载,从而提高模块的加载效率
3.内置模块的加载机制
内置模块是由Node.js官方提供的模块,内置模块的加载优先级最高
例如,require('fs')始终返回内置的fs模块,即使在node_modules目录下有名字相同的包也叫做fs
4.自定义模块的加载机制
使用require()方法加载自定义模块时,必须指定以./或../开头的路径标识符,在加载自定义模块时,如果没有指定./或../这样的路径标识符,则node会把它当做内置模块或第三方模块进行加载
同时,在使用require()导入自定义模块时,如果省略了文件的扩展名,则Node.js会按顺序分别尝试加载以下的文件:
①按照确切的文件名进行加载
②补全.js扩展名进行加载
③补全.json扩展名进行加载
④补全.node扩展名进行加载
⑤如果以上都加在失败,终端报错
5.第三方模块的加载机制
如果传递给require()的模块标识符不是一个内资模块,也没有以./或../开头,则Node.js会从当前模块的父目录开始,尝试从/node_modules文件夹中加载第三方模块
如果没有找到对应的第三方模块,则移动到在上一层目录中,进行加载,直到文件系统的根目录
6.目录作为模块