一、模块分类

  1. 由Node提供的模块,称为核心模块;部分核心模块在Node源代码的编译过程中,编译进了二进制执行文件。在node进程启动时,该部分就直接加载进内存,文件定位和编译执行的步骤可以省略掉,并且在路径分析中优先判断,所以它的加载速度是最快的
  2. 用户编写的模块,成文文件模块;文件模块在运行时动态加载,需要完整的路径分析、文件定位、编译执行过程,速度比核心模块慢。

二、模块规范(CommonJs)

  1. 模块的引用 
    1.  var http = require('math');
    2. 在CommonJs中存在require()方法,这个方法接受模块标识,以此引入一个模块的API到当前的上下文中。
  2. 模块定义
    1. 用exports对象导出当前模块的方法或者变量,并且它是唯一导出的出口。在模块中,还存在module对象,代表模块本身,而exports是module的属性。在node中,一个文件就是一个模块,将方法挂载在exports对象上作为属性即可定义导出的方式。

          

  1. //math.js
  2.  
  3. exports.add = function(){
  4. var sum = 0,
  5. i =0 ,
  6. args = arguments,
  7. l =args.length;
  8. while (i<1){
  9. sum += args[i++];
  10. }
  11. return sum
  12. }
  13.  
  14. //program.js
  15.  
  16. var math = require('math');//引入自定义模块(math.js)
  17.  
  18. exports.increment = function(val){
  19. return math.add(val,1);//使用模块定义的方法
  20. };

3.模块标识

    模块标识就是传递给require()方法的参数,它必须是符合小驼峰命名的字符串,或者以. 、..开头的相对路径,或者绝对路径,可以没有后缀名.js

三、 Node模块实现

实现三步骤

1.路径分析

通过标识符进行模块查找,分几类

  1. 核心模块,如http,fs,path等。
  2. .或..开始的相对路径文件模块。
  3. 以/开始的绝对路径文件模块
  4. 非路径形式的文件模块,如自定义的connect模块。

加载方式和优先级

  • 核心模块的优先级仅此于缓存加载
  • 路径形式的文件模块,require()方法在分析的时候会转为真实路径,并以真实路径作为索引,将编译执行后的结果放到缓存中,以使二次加载时更快。
  • 自定义模块指的是非核心模块,也不是路径行驶的标识符,它是一种特殊的文件模块,可能是文件或包的形式。这类模块查找最费时。
2.文件定位

注意:从缓存加载的优化策略使得二次引入是不需要路径分析、文件定位、编译执行的过程,大大提高了再次加载模块时的效率。

在文件定位的过程中,还要注意一些细节,包括文件扩展名的分析、目录和包的处理。

  • 文件扩展名分析

    • Node会按.js,.json,.node的次序补足扩展名,依次尝试。尝试过程中需要调用fs模块同步阻塞式地判断文件是否存在。因为Node是单线程的,所以这里会引起性能问题。
    • 小诀窍:1.如果是.node,.json文件,在传递给require()的标识符中带上扩展名。2.同步配合缓存
  • 目录分析和包
    • ruquire()通过分析文件扩展名后可能没有查找到对应的文件,但却得到一个目录,Node将把目录当作一个包来处理
    • 在这个过程中,Node在当前目录下查找package.json(CommonJs包规范定义的包描述文件),通过JSON.parse()解析出包描述对象,从中取出main属性指定的文件名来定位,如果指定的文件错误,或没有package.json文件,Node将index当作默认文件名,然后依次查找index.js,index.json,index.node。若还是没有,则抛出查找异常错误。
3.编译执行

每个文件模块都是一个对象,定位到具体文件后,Node就会新建一个模块对象,然后根据路径载入并编译。对于不同的文件扩展名,载入方法不同

  • .js文件。通过fs模块同步读取文件后编译执行。
  • .node文件,用C/C++编写的扩展文件,通过dlopen()方法加载最后编译生成的文件。
  • .json文件,通过fs模块同步读取文件后,用JSON.parse()解析返回结果。
  • 其余扩展名文件。它们都当作.js文件载入。

每个编译成功的模块都会将其文件路劲作为索引缓存在Module._cache对象上,以提高二次引入的性能。

Node学习笔记之模块实现的更多相关文章

  1. node 学习笔记 - Modules 模块加载系统 (1)

    本文同步自我的个人博客:http://www.52cik.com/2015/12/11/learn-node-modules-path.html 用了这么久的 require,但却没有系统的学习过 n ...

  2. node 学习笔记 - Modules 模块加载系统 (2)

    本文同步自我的个人博客:http://www.52cik.com/2015/12/14/learn-node-modules-module.html 上一篇讲了模块是如何被寻找到然后加载进来的,这篇则 ...

  3. node学习笔记6——自定义模块

    自定义模块三大关键词: require——引入模块: exports——单个输出: module——批量输出. 从例子下手: 1.创建module.js: exports.a=22; exports. ...

  4. node学习笔记(二)(ajax方式向node后台提交数据)

    通过ajax向node后台提交数据过程(附手写前后台代码),并总结post与get的区别 POST 前台代码 //CSS简单给点样式 <style> form{ width: 200px; ...

  5. Python学习笔记—itertools模块

    这篇是看wklken的<Python进阶-Itertools模块小结> 学习itertools模块的学习笔记 在看itertools中各函数的源代码时,刚开始还比较轻松,但后面看起来就比较 ...

  6. node学习笔记3——文件操作fs

    文件操作关键字: http('fs') ——  请求 node 里面的 http 模块 readFile ——  读文件,参数包括 文件名,回调函数 writeFile ——  写文件,参数包括 文件 ...

  7. Python学习笔记之模块与包

    一.模块 1.模块的概念 模块这一概念很大程度上是为了解决代码的可重用性而出现的,其实这一概念并没有多复杂,简单来说不过是一个后缀为 .py 的 Python 文件而已 例如,我在某个工作中经常需要打 ...

  8. node学习笔记

    一.准备(github地址) 什么是Javascript? ... Javascript能做什么? ..... 浏览器中的Javascript可以做什么? 操作DOM(增删改查) AJAX/跨域 BO ...

  9. node学习笔记(三)

    //事件驱动events //events是node最重要的模块没有之一,因为node.js本身的架构就是事件式的,而他提供了唯一的接口,所以堪称node.js事件编程的基石; //events几乎被 ...

随机推荐

  1. CentOS7.4+MongoBD3.6.4集群(Shard)部署以及大数据量入库

    前言 mongodb支持自动分片,集群自动的切分数据,做负载均衡.避免上面的分片管理难度.mongodb分片是将集合切合成小块,分散到若干片里面,每个片负责所有数据的一部分.这些块对应用程序来说是透明 ...

  2. 通过代码动态创建IIS站点

    对WebApi进行单元测试时,一般需要一个IIS站点,一般的做法,是通过写一个批处理的bat脚本来实现,其实通过编码,也能实现该功能. 主要有关注三点:应用程序池.Web站点.绑定(协议类型:http ...

  3. 前端模块化方案全解(CommonJS/AMD/CMD/ES6)

    模块化的开发方式可以提高代码复用率,方便进行代码的管理.通常一个文件就是一个模块,有自己的作用域,只向外暴露特定的变量和函数.目前流行的js模块化规范有CommonJS.AMD.CMD以及ES6的模块 ...

  4. HttpSession implements session

    体验 使用HttpSession进行会话管理,完全可以忽略HTTP无状态的事实. HttpSession会话管理原理 使用HttpSession进行会话管理十分方便,让Web应用程序看似可以“记得”浏 ...

  5. zookeeper应用 - leader选举 锁

    模拟leader选举: 1.zookeeper服务器上有一个/leader节点 2.在/leader节点下创建短暂顺序节点/leader/lock-xxxxxxx 3.获取/leader的所有子节点并 ...

  6. redis介绍(6)集群(ruby)

    redis集群: redis集群是高可用的一种体现,让整个redis圈更加稳定,不易出现宕机的情况, redis原理: redis3.0之前是不支持集群的,实现集群要自己去配置实现,很麻烦,在3.0之 ...

  7. A B C D类网络地址

    A类网络地址(红色为网络地址,黑色为主机地址): 下限:0000 0001.0000 0000.0000 0000.0000 0000(1.0.0.0) 上限:0111 1110.1111 1111. ...

  8. flask的g对象

    故名思议我们可以理解这个g对象是一个全局的对象,这个对象存储的是我们这一次请求的所有的信息,只是存储这一次的请求 g:global 1. g对象是专门用来保存用户的数据的.  2. g对象在一次请求中 ...

  9. Python学习---django-debug-tools安装

    [官网]http://django-debug-toolbar.readthedocs.io/en/1.2/installation.html [更多安装参考]http://blog.csdn.net ...

  10. Linux 系统的用户和组详解_【all】

    1.Linux 用户和用户组详解 2.Linux 文件特殊权限详解 3.Linux 文件的读写执行权限的说明 4.Linux 架构之简述企业网站 5.Linux 环境变量设置详解 6.企业生产环境用户 ...