node-学习之路02 commonJS模块

1.commonjs 模块管理

  所有代码都运行在模块作用域,不会污染全局作用域。
  模块可以多次加载,但是只会在第一次加载时运行一次,然后运行结果就被缓存了,以后再加载,就直接读取缓存结果。要想让模块再次运行,必须清除缓存。
  模块加载的顺序,按照其在代码中出现的顺序。

2.module对象

 

2.1 module.exports属性

  module.exports属性表示当前模块对外输出的接口,其他文件加载该模块,实际上就是读取module.exports变量。

2.2 exports变量

  为了方便,Node为每个模块提供一个exports变量,指向module.exports。这等同在每个模块头部,有一行这样的命令:

  

var exports = module.exports;(commonJS隐式做了这个赋值)

2.3 js文件就是一个函数

console.log(arguments)

{ '0': {},
  '1': 
   { [Function: require]
     resolve: { [Function: resolve] paths: [Function: paths] },
     main: 
      Module {
        id: '.',
        exports: {},
        parent: null,
        filename: 'C:\\Users\\liang\\Desktop\\node\\hello.js',
        loaded: false,
        children: [],
        paths: [Array] },
     extensions: { '.js': [Function], '.json': [Function], '.node': [Function] },
     cache: { 'C:\Users\liang\Desktop\node\hello.js': [Object] } },
  '2': 
   Module {
     id: '.',
     exports: {},
     parent: null,
     filename: 'C:\\Users\\liang\\Desktop\\node\\hello.js',
     loaded: false,
     children: [],
     paths: 
      [ 'C:\\Users\\liang\\Desktop\\node\\node_modules',
        'C:\\Users\\liang\\Desktop\\node_modules',
        'C:\\Users\\liang\\node_modules',
        'C:\\Users\\node_modules',
        'C:\\node_modules' ] },
  '3': 'C:\\Users\\liang\\Desktop\\node\\hello.js',
  '4': 'C:\\Users\\liang\\Desktop\\node' }

console.log(arguments.callee)
[Function]
console.log(arguments.length)
5

  (function (exports, require, module, __filename, __dirname){})()

    exports是module.exports的一个引用

    require引用模块后,返回给调用者的是module.exports而不是exports

    exports.xxx,相当于在导出对象上挂属性,该属性对调用模块直接可见

    exports =相当于给exports对象重新赋值,调用模块不能访问exports对象及其属性

    如果此模块是一个类,就应该直接赋值module.exports,这样调用者就是一个类构造器,可以直接new实例

扫描二维码关注公众号,回复: 3467081 查看本文章

    这是两种暴漏的方式,都可以使用

             

              下面是引入方式,及结果

    

    我们可以看成 exports = module.exports = {}  就是exports 指针指向module.exports的对象

             

    关于rfequire 引入的返回值是module.exports

    

 2.4 require 规则

  (1)如果参数字符串以“/”开头,则表示加载的是一个位于绝对路径的模块文件。比如,require('/home/marco/foo.js')将加载/home/marco/foo.js。

  (2)如果参数字符串以“./”开头,则表示加载的是一个位于相对路径(跟当前执行脚本的位置相比)的模块文件。比如,require('./circle')将加载当前脚本同一目录的circle.js。

  (3)如果参数字符串不以“./“或”/“开头,则表示加载的是一个默认提供的核心模块(位于Node的系统安装目录中),或者一个位于各级node_modules目录的已安装模块(全局安装或局部安装)

  (4)如果参数字符串不以“./“或”/“开头,而且是一个路径,比如require('example-module/path/to/file'),则将先找到example-module的位置,然后再以它为参数,找到后续路径。

  (5)如果指定的模块文件没有发现,Node会尝试为文件名添加.js、.json、.node后,再去搜索。.js件会以文本格式的JavaScript脚本文件解析,.json文件会以JSON格式的文本文件解析,.node文件会以编译后的二进制文件解析。

  (6)如果想得到require命令加载的确切文件名,使用require.resolve()方法。

  2.5模块缓存

  第一次加载某个模块时,Node会缓存该模块。以后再加载该模块,就直接从缓存取出该模块的module.exports属性。

  

require('./example.js');require('./example.js').message = "hello";require('./example.js').message// "hello"

  上面代码中,连续三次使用require命令,加载同一个模块。第二次加载的时候,为输出的对象添加了一个message属性。但是第三次加载的时候,这个message属性依然存在,这就证明require命令并没有重新加载模块文件,而是输出了缓存。
  如果想要多次执行某个模块,可以让该模块输出一个函数,然后每次require这个模块的时候,重新执行一下输出的函数。
  所有缓存的模块保存在require.cache之中,如果想删除模块的缓存,可以像下面这样写。

  2.6require.main

require.main === module// true

猜你喜欢

转载自www.cnblogs.com/liangfc/p/9751975.html