专业网站建设品牌,十四年专业建站经验,服务6000+客户--广州京杭网络
免费热线:400-683-0016      微信咨询  |  联系我们

实现javascript模块加载

当前位置:网站建设 > 技术支持
资料来源:网络整理       时间:2023/2/14 0:19:21       共计:3598 浏览

面试官:说说null和undefined的区别

我心想:null即空,额,空即是色,色即是空,皇后大道东又皇后大……吗旦,我在想什么

听朋友说,阿里爸爸有一道这样的面试题:有两个模块A和B,要求加载完A才加载B,说说你的想法。

我第一个想法就是:

moduleA(); moduleB(); 

Bingo!搞定!

屁颠屁颠地去撸一盘lol,撸到一半,感觉不对啊,是不是太简单了?重新打量了打量这个问题,我脑袋一拍想到,其实面试官想问的是:javascript模块加载的实现方法

关于javascript模块加载器,我接触过两种:seajs 和 requirejs。我比较熟悉的是requirejs,所以我按照requirejs的API实现了一个简单的模块加载器。

我没看过requirejs源码,下面所说只是我个人的想法。

如何加载js文件?

我用的是最简单的方法:

var loadJs = function(name) { var script = document.createElement('script'); script.src = name + '.js'; document.body.appendChild(script); }; 

模块加载

模块加载最关键的问题是:如何保证模块执行的顺序。在解决这个问题之前,我们需要注册(记录)模块的相关信息。

如何注册模块?

先准备两个函数require和define,这两个函数功能相似,以define为例子,它接受三个参数:

// define接收三个参数: // - 该模块的名字 // - 依赖数组,包含该模块依赖的其他模块名 // - 回调函数,当模块及其依赖加载完后执行 var define = function(name, depArr, callback) {}; 

该函数接收的三个参数就是模块的相关信息,我们将这些信息记录下来:

var mixJS = { // 存放各个模块的相关信息 modules: {} }, define = function(name, depArr, callback) { // 记录 该模块的信息 mixJS.modules[name] = { depArr: depArr, func: callback }; // loadJs 加载依赖 // …… …… }; 

注册完模块,我们还需要继续加载该模块的依赖,依赖继续运行define函数来注册自身模块,再去加载依赖的依赖……

周而复此,我们能注册到所有模块的信息。接下来的问题就是:几时才开始执行模块?

几时才开始执行模块?

这个问题太简单了不是吗?答案就是:当最后一个模块加载完(注册完)之后,就开始执行模块啊。

好,现在问题变成了:在模块加载过程中,如何知道该模块就是最后一个模块呢?

其实模块加载就像举办宴会。假设每个来宾都可以邀请朋友来参加宴会,而且宴会必须等齐所有来宾才能够开始。如此,宴会主办方无法统计实际会有多少人会到场。于是,主办方派人在入场口记录“待参加人数”,每当有来宾到场,工作人员将“待参加人数”减去此次来宾的人数,并询问他邀请了多少个朋友,将人数增添到“待参加人数”上。再有来宾到场,周而复此……

当主办方发现“待参加人数”为0时,意味着所有来宾都到齐了,此时宴会可以开始了!

根据以上的思路,我们同样可以解决 “模块开始执行时间” 的问题了:

var mixJS = { modules: {}, // 待加载模块的数量,默认值为1,即必须加载main模块 toLoadCount: 1, // 模块执行函数 run: function() {} }, define = function(name, depArr, callback) { // 记录 该模块的信息 // …… …… // loadJs 加载依赖 // …… …… // 更新 待加载模块的数量 // - 该模块已经执行,减1 // - 加上 依赖的数量 mixJS.toLoadCount -= 1; mixJS.toLoadCount += depArr.length; // 如果没有模块等待加载,运行 模块执行函数 if (mixJS.toLoadCount === 0) mixJS.run(); }; 

oh yeah~!接下来到了最关键的问题:“如何按序执行模块呢?”

如何按序执行模块?

首先,为何要按序执行模块?这是因为模块之间存在依赖关系,一个模块的执行过程要用到其依赖模块的返回值,所以必须保证被依赖模块在该模块之前被执行。

解决方法就一句话:在加载模块的过程中,我们将此次模块的依赖压入一个栈中。由于栈的后进先出,所以我们能保证从该栈中取出来的模块只有两种情况:

  • 该模块没有依赖
  • 该模块的依赖已经被执行了

依次从该栈中取出模块,依次地执行模块,就能够保证“按序加载模块”了。实现代码大致如下:

var mixJS = { modules: {}, toLoadCount: 1, // 依赖栈 depStack: [], // 模块执行函数 run: function() { // 1. 按照后进先出原则,取出depStack中的依赖 // 2. 根据依赖的名字,在modules中查找该依赖的回调函数 // 3. 运行该函数并记录下返回值 } }, define = function(name, depArr, callback) { // 记录 该模块的信息 // …… …… // loadJs 加载依赖 // …… …… // 更新 待加载模块的数量 // …… …… // 更新依赖栈 mixJS.depStack = mixJS.depStack.concat(depArr); // 如果没有模块等待加载,运行 模块执行函数 if (mixJS.toLoadCount === 0) mixJS.run(); }; 

wow! 所有问题都解决了!

符合AMD规范

我们还需要一些修改才能使这个模块加载器更符合AMD规范,这样才能加载其他一些库,比如jQuery:

define.amd = true; 

AMD的规范可以参考:

  • AMD规范英文文档
  • AMD规范中文文档
版权说明:
本网站凡注明“广州京杭 原创”的皆为本站原创文章,如需转载请注明出处!
本网转载皆注明出处,遵循行业规范,如发现作品内容版权或其它问题的,请与我们联系处理!
欢迎扫描右侧微信二维码与我们联系。
·上一条:全局CSS的终结 | ·下一条:jQuery.data原理介绍

Copyright © 广州京杭网络科技有限公司 2005-2025 版权所有    粤ICP备16019765号 

广州京杭网络科技有限公司 版权所有