zoukankan      html  css  js  c++  java
  • [ 转]Node.js模块 require和 exports

    什么是模块?

    node.js通过实现CommonJS的Modules/1.0标准引入了模块(module)概念,模块是Node.js的基本组成部分.一个node.js文件就是一个模块,也就是说文件和模块是一一对应的关系.这个文件可以是JavaScript代码,JSON或者编译过的C/C++扩展.

    Node.js的模块分为两类,一类为原生(核心)模块,一类为文件模块。

    在文件模块中,又分为3类模块。这三类文件模块以后缀来区分,Node.js会根据后缀名来决定加载方法。

    • .js。通过fs模块同步读取js文件并编译执行。
    • .node。通过C/C++进行编写的Addon。通过dlopen方法进行加载。
    • .json。读取文件,调用JSON.parse解析加载。

    Node.提供了exports和require两个对象,其中exports是模块公开的接口,require用于从外部获取一个模块接口,即所获取模块的exports对象.

    require查找策略

    原生模块在Node.js源代码编译的时候编译进了二进制执行文件,加载的速度最快。另一类文件模块是动态加载的,加载速度比原生模块慢。但是Node.js对原生模块和文件模块都进行了缓存,于是在第二次require时,是不会有重复开销的。尽管require方法极其简单,但是内部的加载却是十分复杂的,其加载优先级也各自不同。

    nodejs-require

    require方法接受以下几种参数的传递:

    • http、fs、path等,原生模块。
    • ./mod或../mod,相对路径的文件模块。
    • /pathtomodule/mod,绝对路径的文件模块。
    • mod,非原生模块的文件模块。

    当require一个文件模块时,从当前文件目录开始查找node_modules目录;然后依次进入父目录,查找父目录下的node_modules目录;依次迭代,直到根目录下的node_modules目录。

    简而言之,如果require绝对路径的文件,查找时不会去遍历每一个node_modules目录,其速度最快。其余流程如下:

    1. 从module path数组中取出第一个目录作为查找基准。
    2. 直接从目录中查找该文件,如果存在,则结束查找。如果不存在,则进行下一条查找。
    3. 尝试添加.js、.json、.node后缀后查找,如果存在文件,则结束查找。如果不存在,则进行下一条。
    4. 尝试将require的参数作为一个包来进行查找,读取目录下的package.json文件,取得main参数指定的文件。
    5. 尝试查找该文件,如果存在,则结束查找。如果不存在,则进行第3条查找。
    6. 如果继续失败,则取出module path数组中的下一个目录作为基准查找,循环第1至5个步骤。
    7. 如果继续失败,循环第1至6个步骤,直到module path中的最后一个值。
    8. 如果仍然失败,则抛出异常。

    module.exports还是exports

    我们首先通过一个例子来介绍exports的作用.首先新建一个模块calc.js,代码如下:

    var add = function(a,b){
    	return a + b;
    };
    
    var minus = function(a,b){
    	return a - b;
    };

    再新建一个test.js文件,代码如下 ,

    var calc = require("./calc");  
    
    console.log(calc.add(1,2));

    然后我们在Terminal中执行,发出现如下出错.

    /Users/Liuzc/Desktop/node/text.js:3
    console.log(calc.add(1,2));
    ^
    TypeError: Object #<Object> has no method ‘add’
    at Object.<anonymous> (/Users/Liuzc/Desktop/node/text.js:3:18)
    at Module._compile (module.js:456:26)
    at Object.Module._extensions..js (module.js:474:10)
    at Module.load (module.js:356:32)
    at Function.Module._load (module.js:312:12)
    at Function.Module.runMain (module.js:497:10)
    at startup (node.js:119:16)
    at node.js:901:3

    这时我们修改一下calc.js的代码:

    var add = function(a,b){
    	return a + b;
    };
    
    var minus = function(a,b){
    	return a - b;
    };
    
    exports.add = add;
    exports.minus = minus;

    再次在Terminal中执行node test.js时.正确的显示了结果.

    Liuzcs-MacBook-Pro:node Liuzc$ node text.js
    3

    一个模块可以通过module.exports或exports将函数、变量等导出,以使其它JavaScript脚本通过require()函数引入并使用。

    那么,到底应该用module.exports还是用exports呢?我们先看下面的一个例子:

    console.log(this);
    console.log(exports);
    console.log(module.exports);
    
    console.log(this === exports);
    console.log(this === module.exports);
    console.log(exports === module.exports);

    执行结果是:

    {}
    {}
    {}
    true
    true
    true

    也就是说,exports默认和module.exports指向同一个空对象。

    再看一个例子.calc.js中有代码如下 :

    exports.add = function(a,b){
    	return a + b;
    }
    
    module.exports.add = function(a,b){
    	return a - b;
    }

    这时猜猜执行test.js中的如下代码的结果将会是什么?

    var calc = require("./calc");  
    
    console.log(calc.multiply(4,2));

    对, 结果是2, 而不是8.也就是说如果运行时让exports、this和module.exports指向不同的对象,只有module.exports指向的对象才回被导出。module.exports才是真正的接口,exports只不过是它的一个辅助工具。 最终返回给调用的是module.exports而不是exports。 所有的exports收集到的属性和方法,都赋值给了module.exports。当然,这有个前提,就是module.exports本身不具备任何属性和方法。如果,module.exports已经具备一些属性和方法,那么exports收集来的信息将被忽略。

    如果你想你的模块是一个特定的类型就用module.exports。如果你想的模块是一个典型的“实例化对象”就用exports。

  • 相关阅读:
    SQLServer数据库中开启CDC导致“事务日志空间被占满,原因为REPLICATION”的原因分析和解决办法
    译:SQL Server的Missing index DMV的 bug可能会使你失去理智---慎重看待缺失索引DMV中的信息
    SQLServer中间接实现函数索引或者Hash索引
    MySQL缓存分类和配置
    MySQL系统变量配置基础
    MySQL索引统计信息更新相关的参数
    Sql Server优化---统计信息维护策略
    SQL Server 用角色(Role)管理数据库权限
    sp_executesql 或者 EXECUTE 执行动态sql的权限问题
    关于T-SQL重编译那点事,内联函数和表值函数在编译生成执行计划的区别
  • 原文地址:https://www.cnblogs.com/neights/p/4829207.html
Copyright © 2011-2022 走看看