初识commonjs

一,为什么要用commonjs

      在大型项目中,需要根据需求和功能将程序划分为不同的模块。第一有利于多人开发,提高开发效率;第二提高了模块的复用率,清晰明确的框架也能提高程序的可维护性和健壮性。

       但是多人开发也会带来一些问题。

      依赖问题:

             不同的模块之间可能存在依赖问题,如果需要手动指定其加载次序,不仅浪费人力物力,也不利于修改。

      全局污染问题:

            加载次序不同的模块,如果重复定义了相同的变量,一定会引发无法预计的后果!

      模块化开发是一种很好的思想,但是需要有一定的规范加以约束!

      在服务器端,采用了commonjs,在浏览器端,有AMDCMD之分。

      今天先了解一下commonjs,一步步吃成胖子!

二,准备工作

      安装nodejsnpm

三,module对象

      commonjs认为每个文件都是一个模块,每个模块都是一个作用域,通过require其他模块来解决模块的加载次序问题,通过exports来暴露接口,解决命名冲突问题。

      每个模块内部都有一个module对象,指向当前模块,具有如下属性:

            module.id                           模块的识别符,通常是带有绝对路径的模块文件名

            module.filename              模块的文件名,带有绝对路径

            module.loaded                 boolean,标志模块是否已经加载

            module.parent                  object,表示调用该模块的模块

            module.children               array,表示该模块需要用到的模块

            module.exports                模块暴露的属性和方法

四,require

      当我们键入 > var http = require(‘http’)commonjs就帮我们引入了http这个模块,看上去很简单,就是加载了一个模块,其实走了下面一段流程:

4.1查看缓存

      可以多次require一个模块,但是除了第一次是真正的下载、执行一个js文件,然后返回其module.exports对象,并存储起来,之后的require都是到直接获取相应的缓存。

       

                        图4-1

      

                  图4-2

       

                  图4-3

      可以发现require(‘./sub.js’)其实只执行了一次,与之后的require(‘./sub.js’)共享同一个缓存。

      commonjs将模块存储在require.cache,修改main.js的内容,通过 > delete require.cache[ moduleName ]来清除缓存

       

                  图4-4

      

                  图4-5

      观察打印的key值,可以发现commonjs将绝对路径作为模块名,来标识模块的缓存

4.2加载模块

      当在缓存中找不到相应的模块,就先创建相应的module实例存入缓存,然后加载相应的模块。

  4.2.1参数的解析

         /:表示加载绝对路径的模块文件

         ./:表示加载相对路径的模块文件

         默认提供的核心模块:先检索node的系统安装目录,在检索各级的node_modules

             比如/home/user/project/foo.js执行了require('bar.js'),则会顺序查找:

                 /user/local/lib/node/bar.js

                 /home/user/project/node_modules/bar.js

                 /home/user/node_modules/bar.js

                 /home/node_modules/bar.js

                 /node_mudules/bar.js

         纯路径:先搜索该路径的package.json,然后加载其中main指定的js文件,

              如果没有main或者package.json

             加载该目录的index.js或者index.node文件

         如果找不到相应的模块,自动添加后缀名:jsjsonnode(以编译后的二进制文件解析)

  4.2.2同步加载

         require是同步加载的,按照在代码中出现的次序顺序加载。

         即读入、执行一个js文件,然后返回其module.exports对象再继续往下执行;

         如果require一个不存在的文件,则报错。

  4.2.3嵌套加载跟循环加载

         如果存在嵌套的require,则外部的require等待内部的require结束在继续执行。

         如果存在循环require,则会因为缓存的机制获得不完全加载的版本

 

                  图4-6

 

                  图4-7

 

                  图4-8

 

                  图4-9

4.3加载失败

   

                图4-10

 

               图4-11

 

               图4-12

     可见即使没执行完相应的js文件,也会预先在缓存中存储module实例(存在main.js)。

     当加载失败时,从require.cache中删除预定义的module实例(删除sub.js

4.4加载成功

     返回module.exports对象。

五,exports

 5.1 exportsmodule.exports

     每个模块默认定义了var exports = module.exports,通过向exports添加方法属性,等同于添加到module.exports上。

     注意,module.exportsexports指向同一块内存

     如果对exports整体赋值,相当于改变exports的指向,则之后的任何修改都不会影响module.exports了。

     如果对module.exports整体赋值,相当于改变module.exports的指向,则module.exports和exports分道扬镳,互不干涉了。

所以如果最后只导出一个函数,只能通过赋值给module.exports来导出

 5.2 注意exports的闭包问题

    

               图5-1

     按照我的理解,这种所谓的影响不到有两个原因:

          1counter是原始数据类型

          2,闭包。

     原因这里先不分析,之后会写写作用域链跟闭包,读者可以到时参考后自行分析。

     修改一

 

               图5-2

 

               图5-3

 

               图5-4

     思路:利用闭包保存共同的引用

    修改二

    

               图5-5

 

               图5-6

     思路:在函数运行时才确定其上下文环境

六,其他

  6.1 全局变量

     每个模块都是一个作用域,内部的变量在其他地方不可见,

     可以将变量设置为global的属性改变其可见性

  6.2 require.main === module

     返回true的话,标识着模块是直接执行

     否则为被调用执行

  6.3 IIFE(Immediately-Invoked Function Expressions)

     立即执行函数,一般减少全局作用域的污染。

     之前经常看到将jQuery等常用库作为参数传递进来,今天结合作用域链终于发现其好处了:将原本需要到全局作用域里查找的东西“放入”当前的作用域,如果在嵌套很深的情况下,能减少很多无谓的搜索!

  6.4 参考

         http://javascript.ruanyifeng.com/nodejs/module.html#toc0

  • 0
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值