深入探寻seajs的模块化与加载方式


Posted in Javascript onApril 14, 2015

由于一直在使用,所以了解了下seajs的源代码。这里是我对下面几个问题的理解:

1、seajs的require(XXX)的方法是怎样实现模块加载的?

2、为什么需要预加载?

3、为什么需要构建工具?

4、构建前后的代码究竟有些什么区别,为什么要这么做?

问题1: seajs的require(XXX)的方法是怎样实现模块加载的?

代码逻辑比较绕,对源代码的理解放在文章的末尾,这里先简单梳理下模块加载的逻辑:

1、从seajs.use方法入口,开始加载use到的模块。

2、use到的模块这时mod缓存当中一定是不存在的。seajs创建一个新的mod,赋予一些初始的状态。

3、执行mod.load方法

4、一堆逻辑之后走到seajs.request方法,请求模块文件。模块加载完成之后,执行define方法。

5、define方法分析提取模块的依赖模块,保存起来。缓存factory但不执行。

6、模块的依赖模块再被加载,如果继续有依赖模块,则继续加载。直至所有被依赖的模块都加载完毕。

7、所有的模块加载完毕之后,执行use方法的callback.

8、模块内部逻辑从callback开始执行。require方法在这个过程当中才被执行。

问题2:为什么需要预加载?

我们看到seajs.use方法实际上是在所有依赖模块都加载完了之后才执行callback。可以理解成在业务逻辑代码在执行之前,必须先预加载所有被依赖的模块代码。那么为什么是一个这样必须先做预加载的逻辑?

答案在于逻辑代码里面引用其他模块方法的这个require方法的执行方法:

var mod = require(id);

这个语法决定了mod的取得是个同步执行的过程,如果模块代码在此之前没有被预加载的话,就只能采用异步加载回调的方法来实现了,那么整个seajs的执行逻辑将完全会是另一个样子。因为异步你会搞不懂模块的执行顺序,逻辑会变的难以掌控。

问题3:为什么需要构建工具?

可以看到没有构建前各个依赖模块都是单独加载的。这会产生过多的模块请求,对于页面的加载性能是不利的。构建工具本质上就是为了解决模块合并加载的问题。

问题4:构建前后的代码究竟有些什么区别,为什么要这么做?

构建工具究竟做了些什么。我们说它本质上是为了解决代码合并加载的问题,那么它所做的只是简单的将各个模块文件合并成一个文件?

当然不是。测试一下,你如果只是简单把几个模块文件合并到一个文件以后,会发现这个文件根本没有办法正常执行。

原因在于define方法的实现。

seajs是推崇定义模块的时候只在define方法传入factory参数的。回顾define方法内部,当没有传入id(姑且等同于模块的url)时,会通过getCurrentScript()方法去取得当前正在执行的这个模块文件的url路径,然后把这个路径作为键值与模块本身一起缓存到cachedMods。这里很关键的一点是,整个seajs内部的这个模块缓存机制其实是依赖每个模块的url来做缓存的键值。require(id)方法,归根结底也是通过url键值到。require(id)方法,归根结底也是通过url键值到cachedMods里面去找相应的模块。这个键值不能重复不能出错,不然模块的对应关系就混乱了。如果把a、b、c几个模块文件简单合并到一个目标文件x之后,getCurrentScript()只能获取到x的路径,三个模块的键值就没法做出区别了,执行肯定出错。

所以如果要把几个模块文件合并在一起,就必须为每个模块明确uri。也就是define方法必须都传入id参数。当id传入的时候,seajs会将这个id转换为url用作缓存的键值。

如果只传id和factory,也就是 define(id, factory),那么deps = undefined,define方法就会去执行parseDependencies(factory.toString())方法提取factory里面的依赖模块,后续会走到解析模块路径,线上单独加载各个模块的逻辑里面去,这个时候就失去了合并加载的意义了。

所以合并加载,define方法必须正确的传入id,deps,factory三个参数才能正确执行。

seajs 所谓CMD的模块定义方法,是提倡大家写模块的阶段都只传factory一个参数的,其他两个参数在后期代码构建的阶段来生成。上面解释了为什么这两参数在构建后是必须的。

至于为什么提倡定义模块的时候只传factory,我看主要是因为手工传入的id和deps参数,极易出错,不便维护。工具可以提高效率并保证参数的正确。

附: 对seajs 主要代码逻辑的理解。

说明:源代码版本是Sea.js 2.3.0

1、先看看define方法做了些什么

Module.define = function (id, deps, factory)

define方法的时候,支持三个参数。其中id,deps是选填的。factory必须。代码里面通过以下逻辑来控制:

深入探寻seajs的模块化与加载方式

但其实deps是必须的,因为seajs必须知道每个模块依赖了哪些模块,不然无法执行加载。

所以,当factory是函数,并且deps没有被主动传入的时候,就需要使用parseDependencies方法来分析出factory当中的依赖模块了。

深入探寻seajs的模块化与加载方式

parseDependencies方法做的事情主要就是用一个正则表达式把函数体里面所有require(XXX)里面的XXX提取出来,这也就是这个函数依赖到的所有模块了。

深入探寻seajs的模块化与加载方式

方法本身不复杂,但是这个正则表达式不简单:

分析完deps之后,将模块定义存入缓存:

深入探寻seajs的模块化与加载方式

注意,我们会发现define方法纯粹只是分析模块、存储模块,并没有执行模块。

2、真正执行模块,是在require方法里面。我们接下来看require。

深入探寻seajs的模块化与加载方式

深入探寻seajs的模块化与加载方式

简而言之require方法就是根据id在define定义存储的模块缓存中找到相应的模块,并执行它,获得模块定义返回的方法:

深入探寻seajs的模块化与加载方式

整个这个大步骤中,有一个很关键的步骤,有必要细说:

Module.get(require.resolve(id))。

require一个模块的时候,首先要找到这个模块。 Module.get方法就起这个作用。

深入探寻seajs的模块化与加载方式

cachedMods里面没有的话,就创建一个新的Module并缓存到cachedMods里面:

深入探寻seajs的模块化与加载方式

define和rquire方法这样看来不算复杂。seajs主要还是模块加载的逻辑有点复杂。

3、seajs真正执行的入口,是use方法:

深入探寻seajs的模块化与加载方式

通过use方法,从这里的ids开始触发模块的加载和执行。

深入探寻seajs的模块化与加载方式

可以看到加载的关键点在mod.load方法。

load方法代码有点长,其中的主要逻辑是:判断mod的当前状态是否为已加载或者加载中。

深入探寻seajs的模块化与加载方式

在Module的舒适化函数中,我们可以看到status默认值是0.

深入探寻seajs的模块化与加载方式

所以没有加载过的新模块,到这里都是: mod.status = STATUS.LOADING 状态设置为加载中,并执行后续加载逻辑。

接来下是获取模块的依赖urls

深入探寻seajs的模块化与加载方式

mod.resolve方法:

深入探寻seajs的模块化与加载方式

Module.resolve方法本质上就是把相对路径、配置的path、别名等转换成一个绝对路径。不贴代码了。

更新模块加载状态。

深入探寻seajs的模块化与加载方式

加载模块的逻辑:

深入探寻seajs的模块化与加载方式

主要是m.fetch方法,里面其他逻辑这里略过。

深入探寻seajs的模块化与加载方式

可以看到 seajs.request最终会去执行模块文件的加载:

深入探寻seajs的模块化与加载方式

当所有依赖模块加载完了之后,执行mod的onload方法

深入探寻seajs的模块化与加载方式

这里是 mod.onload()方法

深入探寻seajs的模块化与加载方式

到此,seajs的核心逻辑就差不多都看到了。供参考,有理解不到位或者表达不准确的地方,欢迎一起探讨。

以上所述就是本文的全部内容了,希望大家能够喜欢。

Javascript 相关文章推荐
js压缩利器
Feb 20 Javascript
地震发生中逃生十大法则
May 12 Javascript
jquery中获得$.ajax()事件返回的值并添加事件的方法
Apr 15 Javascript
javascript中使用css需要注意的地方小结
Sep 01 Javascript
javascript内存管理详细解析
Nov 11 Javascript
jquery中$each()方法的使用指南
Apr 30 Javascript
jquery操作angularjs对象
Jun 26 Javascript
JS实现的自定义右键菜单实例二则
Sep 01 Javascript
基于JavaScript实现手机短信按钮倒计时(超简单)
Dec 30 Javascript
Angular懒加载机制刷新后无法回退的快速解决方法
Aug 30 Javascript
在vue 中使用 less的教程详解
Sep 26 Javascript
Javascript基于OOP实实现探测器功能代码实例
Aug 26 Javascript
javascript数组去重的方法汇总
Apr 14 #Javascript
JavaScript字符串常用类使用方法汇总
Apr 14 #Javascript
JavaScript 表单处理实现代码
Apr 13 #Javascript
JavaScript 事件绑定及深入
Apr 13 #Javascript
JavaScript 事件对象介绍
Apr 13 #Javascript
JavaScript 事件入门知识
Apr 13 #Javascript
JavaScript 动态加载脚本和样式的方法
Apr 13 #Javascript
You might like
php循环输出数据库内容的代码
2008/05/24 PHP
PHPExcel简单读取excel文件示例
2016/05/26 PHP
laravel创建类似ThinPHP中functions.php的全局函数
2016/11/26 PHP
jquery 指南/入门基础
2007/11/30 Javascript
Ajax+Json 级联菜单实现代码
2009/10/27 Javascript
对象无length属性时IE6/IE7中无法将其转换成伪数组(ArrayLike)
2011/07/31 Javascript
通过JavaScript控制字体大小的代码
2011/10/04 Javascript
用JavaScript修改CSS属性的代码
2013/05/06 Javascript
如何使用jQuery Draggable和Droppable实现拖拽功能
2013/07/05 Javascript
js hover 定时器(实例代码)
2013/11/12 Javascript
jquery 取子节点及当前节点属性值的方法
2014/08/24 Javascript
浅谈JavaScript数据类型
2015/03/03 Javascript
浅谈js多维数组和hash数组定义和使用
2016/07/27 Javascript
Bootstrap精简教程中秋大放送
2016/09/15 Javascript
ajax级联菜单实现方法实例分析
2016/11/28 Javascript
Json按某个键的值进行排序
2016/12/22 Javascript
vue 实现全选全不选的示例代码
2018/03/29 Javascript
vue组件实现弹出框点击显示隐藏效果
2020/10/26 Javascript
VueJs里利用CryptoJs实现加密及解密的方法示例
2019/04/29 Javascript
微信小程序实现分页加载效果
2020/11/19 Javascript
简单介绍Python中的几种数据类型
2016/01/02 Python
Python实现字符串匹配算法代码示例
2017/12/05 Python
Python爬虫抓取代理IP并检验可用性的实例
2018/05/07 Python
numpy matrix和array的乘和加实例
2018/06/28 Python
PyQt5 QTableView设置某一列不可编辑的方法
2019/06/25 Python
python 判断字符串中是否含有汉字或非汉字的实例
2019/07/15 Python
Python企业编码生成系统之主程序模块设计详解
2019/07/26 Python
Python开发企业微信机器人每天定时发消息实例
2020/03/17 Python
Koral官方网站:女性时尚运动服
2019/04/10 全球购物
巴西最大的运动品牌:Olympikus
2020/07/14 全球购物
入党积极分子自我鉴定范文
2014/03/25 职场文书
调解协议书
2014/04/16 职场文书
学前班评语大全
2014/05/04 职场文书
2016年中秋节慰问信
2015/12/01 职场文书
带你了解CSS基础知识,样式
2021/07/21 HTML / CSS
HTML中实现音乐或视频自动播放案例详解
2022/05/30 HTML / CSS