JS 异步插件系统方案研究

广告位招租
扫码页面底部二维码联系

在我们开发设计的 js 项目中,我们经常【作者:唐霜】本文作者:唐霜,转载请注明出处。会设计一套插件系统,用以可选的加载某些功【转载请注明来源】【原创不易,请尊重版权】能模块,在使用时,需要某个功能,就加载对【关注微信公众号:wwwtangshuangnet】本文版权归作者所有,未经授权不得转载。应的插件。以 jquery 举例,它的插【本文首发于唐霜的博客】【版权所有,侵权必究】件生态极其丰富,要使用一个 jquery著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net 插件非常简单,只需要引入 jquery【关注微信公众号:wwwtangshuangnet】【本文受版权保护】 再引入 jquery 插件,就可以利用【转载请注明来源】【版权所有】唐霜 www.tangshuang.net插件提供的功能了。插件提供的功能可能是增【原创不易,请尊重版权】【本文首发于唐霜的博客】加了一个事件绑定,或者提供了新的 $  【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】方法,等等。总之,你不需要插件,你就不需【作者:唐霜】著作权归作者所有,禁止商业用途转载。要加载该插件,而你一旦加载一个插件,就必著作权归作者所有,禁止商业用途转载。【转载请注明来源】须按照插件作者提供的文档来写代码。

未经授权,禁止复制转载。未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。

我们这篇文章并不详细阐述,什么样的一个插未经授权,禁止复制转载。转载请注明出处:www.tangshuang.net件系统是好的插件系统(起码我认为 jqu【转载请注明来源】【原创不易,请尊重版权】ery 的插件系统就非常优秀)。我们这篇【版权所有,侵权必究】【转载请注明来源】文章要来阐述一个插件的异步方案。

转载请注明出处:www.tangshua【本文首发于唐霜的博客】【原创内容,转载请注明出处】ng.net【关注微信公众号:wwwtangshua未经授权,禁止复制转载。【转载请注明来源】ngnet】转载请注明出处:www.tangshua【版权所有,侵权必究】【原创内容,转载请注明出处】ng.net

什么是异步插件?

首先,我们要指明,怎样的一种行为方式是异【原创不易,请尊重版权】未经授权,禁止复制转载。步插件。以及,我们为什么需要异步插件。

【原创不易,请尊重版权】【原创内容,转载请注明出处】【关注微信公众号:wwwtangshua【作者:唐霜】本文版权归作者所有,未经授权不得转载。ngnet】【关注微信公众号:wwwtangshua本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。ngnet】

像 jquery 加载插件那样未尝不可,本文作者:唐霜,转载请注明出处。【本文受版权保护】但直接从 cdn 引入插件会阻塞浏览器渲【转载请注明来源】【本文首发于唐霜的博客】染。而且 2020 年的前端,避不开构建【原创不易,请尊重版权】转载请注明出处:www.tangshuang.net、打包等等工程化过程。异步插件解决减小 【未经授权禁止转载】【未经授权禁止转载】bundle 文件的问题,同时,由于异步【原创内容,转载请注明出处】【转载请注明来源】加载,因此,也不会阻塞界面渲染。带来的问【版权所有,侵权必究】【版权所有,侵权必究】题就比较复杂,如何异步渲染,并且同时能够【作者:唐霜】【原创内容,转载请注明出处】按需要的顺序执行。

【版权所有,侵权必究】原创内容,盗版必究。【作者:唐霜】

对于插件的使用者而言,更关注这个异步插件【版权所有,侵权必究】转载请注明出处:www.tangshuang.net的 cdn 地址,以及插件的使用方法。我未经授权,禁止复制转载。【原创内容,转载请注明出处】们用代码来看下具体的情况是怎么样的。来看【本文受版权保护】【关注微信公众号:wwwtangshuangnet】下一个同步插件:

【访问 www.tangshuang.n【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.netet 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。【转载请注明来源】
<script src="my-lib.min.js"></script>
<script src="my-lib.some-plugin.min.js"></script>
<script>
const myLib = new MyLib()
// some 这个方法是加载了 some-plugin 之后才有的
myLib.some()
</script>

同步插件的问题在于会阻塞渲染,在加载插件【本文受版权保护】著作权归作者所有,禁止商业用途转载。的时候,界面渲染会阻塞,而且如果插件文件本文版权归作者所有,未经授权不得转载。【本文受版权保护】执行过程中出了问题,报错了,还会导致所有本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。的脚本都无法执行,最终导致应用坏死。而我【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。们来看下异步插件:

【访问 www.tangshuang.n【本文受版权保护】【未经授权禁止转载】et 获取更多精彩内容】转载请注明出处:www.tangshua本文作者:唐霜,转载请注明出处。【转载请注明来源】ng.net【转载请注明来源】
<script src="my-lib.min.js"></script>
<script>
MyLib.registerPlugin('some-plugin', 'http://cdn.xxx.com/some-plugin.min.js') // 这句话将会自动去 cdn 异步加载插件

const myLib = new MyLib()
myLib.usePlugin('some-plugin', function() {
  // 异步插件由于异步加载,所以不能立即在程序中去调用 some 方法,而是必须在插件加载完成之后才能做这件事
  this.some()
})
</script>

异步插件会有一些局限性。包括但不限于:

著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshua【未经授权禁止转载】未经授权,禁止复制转载。ngnet】
  • 由于是异步加载的,所以实例化的时候,插件原创内容,盗版必究。本文作者:唐霜,转载请注明出处。可能还没有加载呢,所以插件所提供的功能都本文版权归作者所有,未经授权不得转载。【转载请注明来源】还没有加载完成,因此,在编程上和同步调用【本文受版权保护】【未经授权禁止转载】会有一些不同
  • 【版权所有】唐霜 www.tangshu【访问 www.tangshuang.net 获取更多精彩内容】【原创不易,请尊重版权】ang.net【版权所有】唐霜 www.tangshu本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。ang.net
  • 如果一个项目,需要通过异步的方式加载插件【作者:唐霜】转载请注明出处:www.tangshuang.net,那么项目本身的代码会变得更复杂
  • 【转载请注明来源】本文作者:唐霜,转载请注明出处。
  • 异步插件存在依赖关系怎么办?例如 A 插本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net件依赖 B 插件,怎么解决执行顺序问题?
  • 【访问 www.tangshuang.n【未经授权禁止转载】【关注微信公众号:wwwtangshuangnet】et 获取更多精彩内容】【关注微信公众号:wwwtangshua转载请注明出处:www.tangshuang.net转载请注明出处:www.tangshuang.netngnet】本文作者:唐霜,转载请注明出处。

对于插件开发者的心智是一个挑战,我们需要【原创内容,转载请注明出处】【访问 www.tangshuang.net 获取更多精彩内容】找到更好的异步加载方案,和写代码的正确方【关注微信公众号:wwwtangshuangnet】【原创内容,转载请注明出处】式,这样才能保证开发者用最简单的方式开发本文版权归作者所有,未经授权不得转载。【原创不易,请尊重版权】插件。所以本文接下来,就要探讨一下,怎样【转载请注明来源】【版权所有】唐霜 www.tangshuang.net的 js 插件系统的异步加载方案更优秀,【原创不易,请尊重版权】原创内容,盗版必究。更值得我们推崇。

【版权所有,侵权必究】【访问 www.tangshuang.n未经授权,禁止复制转载。【本文受版权保护】et 获取更多精彩内容】原创内容,盗版必究。

异步插件加载方式

我们将异步加载插件分解为两个问题:1)如原创内容,盗版必究。【原创内容,转载请注明出处】何异步加载 js 脚本?2)如何在该脚本著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net中定义插件?

原创内容,盗版必究。【版权所有】唐霜 www.tangshu本文作者:唐霜,转载请注明出处。【本文受版权保护】ang.net【关注微信公众号:wwwtangshua著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。ngnet】

异步加载脚本

异步加载脚本的方式有很多。总结起来主要分本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。为 3 大类:一类是通过创建 scrip本文作者:唐霜,转载请注明出处。【转载请注明来源】t DOM 实例的 src 实现脚本加载本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】;二类是通过 xhr 获取脚本内容,再在【作者:唐霜】著作权归作者所有,禁止商业用途转载。当前环境中运行脚本内容;三类是通过创建 【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.netiframe 来实现同于脚本运行。这些知【本文受版权保护】【本文首发于唐霜的博客】识可以通过搜索引擎来学习。

未经授权,禁止复制转载。【版权所有】唐霜 www.tangshu【版权所有】唐霜 www.tangshuang.net【关注微信公众号:wwwtangshuangnet】ang.net

我们应该如何选择呢?我们想到的原则是,既【未经授权禁止转载】转载请注明出处:www.tangshuang.net要保证插件系统本身的代码量不要太大,也要【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】保证最终插件的使用者代码写起来不别扭(学【版权所有,侵权必究】原创内容,盗版必究。习成本低)。之所以带来学习成本,最大的问原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。题在于“异步”所带来的副作用。插件脚本在【访问 www.tangshuang.net 获取更多精彩内容】【原创不易,请尊重版权】加载完时,我们很有可能已经完成自己类库的【本文首发于唐霜的博客】【原创不易,请尊重版权】实例化,甚至有些情况下已经做了很多事。然未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net后,在做这些事情的时候,我们可能会忽略插【访问 www.tangshuang.net 获取更多精彩内容】【转载请注明来源】件还没有加载完。而另一方面,如果等到插件【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。加载完再执行所有代码,那会使得实例化过程【未经授权禁止转载】【转载请注明来源】变得非常漫长,导致错过一些重要时间节点上本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】的操作。

【作者:唐霜】本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.n未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.netet 获取更多精彩内容】转载请注明出处:www.tangshua未经授权,禁止复制转载。未经授权,禁止复制转载。ng.net

在我自己的实践中,我选择了通过 scri转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。pt.src 来加载脚本的方式,这种方式【访问 www.tangshuang.net 获取更多精彩内容】转载请注明出处:www.tangshuang.net中,需要设置 script.async 转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】属性为 true,甚至为了兼容,同时设置著作权归作者所有,禁止商业用途转载。【原创内容,转载请注明出处】 script.defer 属性为 tr本文作者:唐霜,转载请注明出处。【未经授权禁止转载】ue。在 onload 中,通知插件系统【本文受版权保护】本文版权归作者所有,未经授权不得转载。,该插件加载好了,需要做一些处理。

未经授权,禁止复制转载。【版权所有,侵权必究】转载请注明出处:www.tangshua【未经授权禁止转载】【版权所有,侵权必究】ng.net【作者:唐霜】

插件的定义

作为插件,而非类库的主要部分,它只是将自著作权归作者所有,禁止商业用途转载。【本文受版权保护】己的功能附加到主库中去。如何将插件插入主【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。库的功能列表中呢?这需要一个合理的设计。

原创内容,盗版必究。原创内容,盗版必究。

实际上,我们需要很简单的一个操作。例如 未经授权,禁止复制转载。【本文受版权保护】MyLib.addPlugin(plug原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。in) 即可,其中的操作由开发者自己去想【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。象。现在的问题是,作为插件的开发者,我们【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。应该怎么去定义一个插件?在异步加载过程中【本文首发于唐霜的博客】【访问 www.tangshuang.net 获取更多精彩内容】,我们可能会有一些需求,例如,我们当前这【本文首发于唐霜的博客】【原创内容,转载请注明出处】个插件需要依赖另外一个插件,但是由于插件著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net是异步加载的,所以对于当前这个插件而言,【作者:唐霜】【作者:唐霜】并不能保证被依赖的插件已经加载好了。所以原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。,我们需要有一个依赖列表,用来表示自己所本文作者:唐霜,转载请注明出处。原创内容,盗版必究。依赖的其他插件。在真正运行的时候,插件系转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。统,应该保证,该插件的依赖被运行之后,才【原创内容,转载请注明出处】转载请注明出处:www.tangshuang.net运行当前这个插件。

著作权归作者所有,禁止商业用途转载。未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。

既然提到了“依赖”,那就不得不说新晋的前本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】端开发者不知道的 AMD 模块加载规范。本文作者:唐霜,转载请注明出处。【未经授权禁止转载】遵循 AMD 的 require.js 本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】曾经流行一时,它通过约定模块加载路径,从【未经授权禁止转载】【本文受版权保护】而使得加载的内容可以根据需要实现 laz【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。yload(这个词大家应该都懂)。而 A【版权所有,侵权必究】原创内容,盗版必究。MD 规范中,规定了一个模块定义的方法,【本文首发于唐霜的博客】原创内容,盗版必究。最主要的用法是

转载请注明出处:www.tangshua【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。ng.net转载请注明出处:www.tangshua转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.netng.net
define(name: string, dependencies: string[], function(...deps: any[]): any {
  // ...
  return exports
})

通过这样一个 define 函数来定义一【未经授权禁止转载】【版权所有】唐霜 www.tangshuang.net个模块,其中 define 的第二个参数【作者:唐霜】【版权所有】唐霜 www.tangshuang.net,就是告诉 require.js 当前这未经授权,禁止复制转载。【关注微信公众号:wwwtangshuangnet】个模块需要依赖哪些模块,在运行当前这个模【原创不易,请尊重版权】【版权所有】唐霜 www.tangshuang.net块之前,要确保 dependencies本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。 这些模块已经运行过了。对于 requi【本文受版权保护】本文作者:唐霜,转载请注明出处。re.js 而言,在运行当前这个模块时,【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】会自动检查依赖,如果依赖没有准备好,re【原创不易,请尊重版权】转载请注明出处:www.tangshuang.netquired.js 会按照自己的模块加载著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。机制,先加载被依赖的模块(加载模块脚本,【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】同时运行模块函数),然后才运行当前这个模本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。块。而在加载依赖时,模块会并行加载和运行原创内容,盗版必究。原创内容,盗版必究。(当然,如果依赖内部还有依赖,它也会安装【关注微信公众号:wwwtangshuangnet】【本文受版权保护】依赖顺序进行执行,但可以肯定的是,模块脚【本文受版权保护】【关注微信公众号:wwwtangshuangnet】本的加载是并行的)。

原创内容,盗版必究。原创内容,盗版必究。转载请注明出处:www.tangshua原创内容,盗版必究。原创内容,盗版必究。ng.net本文版权归作者所有,未经授权不得转载。

AMD 方案不能说最好,但是却非常适用我【本文受版权保护】著作权归作者所有,禁止商业用途转载。所面临的问题——如何高效合理的加载异步插本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。件。

【版权所有】唐霜 www.tangshu【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。ang.net著作权归作者所有,禁止商业用途转载。

因此,最终,我选择了类似 AMD 这样的【转载请注明来源】【本文受版权保护】模块定义方案来定义我们的插件:

转载请注明出处:www.tangshua【原创内容,转载请注明出处】原创内容,盗版必究。ng.net【原创内容,转载请注明出处】著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。
MyLib.defineAsyncPlugin('some', [], function() {
  // 和 AMD 不同,插件的模块函数并非全局执行,而是针对每一个 MyLib 实例运行,所以,在本函数内,插件作者可以调用 this 来指向 MyLib 的实例
  this.some = function() {
    console.log('some is invoked')
  }
})

这样,对于一个 MyLib 的实例而言,原创内容,盗版必究。【转载请注明来源】就拥有了一个 some 方法。

原创内容,盗版必究。【本文首发于唐霜的博客】【关注微信公众号:wwwtangshua【本文受版权保护】原创内容,盗版必究。ngnet】

这里需要有一个更深入的理解。我们的插件虽本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。然在 MyLib 上定义,但是定义的时候转载请注明出处:www.tangshuang.net原创内容,盗版必究。,并不执行插件函数,而是被记录在 MyL转载请注明出处:www.tangshuang.net【未经授权禁止转载】ib 内部,真正拥有插件能力的,是 My【原创不易,请尊重版权】【本文首发于唐霜的博客】Lib 的实例 myLib。这样设计和 转载请注明出处:www.tangshuang.net【原创不易,请尊重版权】require.js 的巨大差别在于,r【本文首发于唐霜的博客】【转载请注明来源】equire.js 是模块系统,本身就是未经授权,禁止复制转载。【本文受版权保护】在全局执行,每一个模块在加载的时候,就需转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。要执行模块函数,而插件则不同,插件是要提【本文受版权保护】【关注微信公众号:wwwtangshuangnet】供功能,而这些功能,往往都是针对实例,因【原创内容,转载请注明出处】【版权所有,侵权必究】此,它不是全局执行,而是在实例化时执行。

【本文首发于唐霜的博客】著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshua转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。ngnet】

定义一个插件只是提供了它的定义,并不立即【转载请注明来源】未经授权,禁止复制转载。运行它。真正的运行是在实例中,这样才能保【本文首发于唐霜的博客】转载请注明出处:www.tangshuang.net证实例拥有插件提供的方法。

【版权所有】唐霜 www.tangshu【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。ang.net【版权所有】唐霜 www.tangshu本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。ang.net【版权所有】唐霜 www.tangshu【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。ang.net

异步插件的注册和使用

和 jquery 插件一样,我们要先注册本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。插件,然后才能使用它。但是和同步加载插件【版权所有】唐霜 www.tangshuang.net未经授权,禁止复制转载。不同,我们不能马上加载和运行异步插件文件【本文首发于唐霜的博客】转载请注明出处:www.tangshuang.net,这样会阻塞进程,达不到我们要的效果。我【转载请注明来源】转载请注明出处:www.tangshuang.net们得使用一种合理的方法来注册插件,这样技原创内容,盗版必究。原创内容,盗版必究。能不阻塞界面渲染,同时又能得到想要的效果原创内容,盗版必究。本文作者:唐霜,转载请注明出处。

【本文受版权保护】未经授权,禁止复制转载。未经授权,禁止复制转载。【本文首发于唐霜的博客】

插件的注册

为什么要先注册再使用呢?为什么不直接在使【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】用的时候自动注册呢?就像 require著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshuangnet】.js 一样,使用的时候自己去加载,加载转载请注明出处:www.tangshuang.net【作者:唐霜】完马上使用。这样做未尝不可以,例如,我们【原创不易,请尊重版权】【关注微信公众号:wwwtangshuangnet】用一段伪代码来表示:

【访问 www.tangshuang.n【作者:唐霜】【原创不易,请尊重版权】et 获取更多精彩内容】【作者:唐霜】本文作者:唐霜,转载请注明出处。
const myLib = new MyLib()
myLib.useAsyncPlugin(url).then(function() {
  this.some()
})

这样的代码非常简洁,甚至你会觉得这才是正【关注微信公众号:wwwtangshuangnet】转载请注明出处:www.tangshuang.net解。

【原创不易,请尊重版权】【访问 www.tangshuang.n【访问 www.tangshuang.net 获取更多精彩内容】【本文受版权保护】et 获取更多精彩内容】【访问 www.tangshuang.n【版权所有,侵权必究】【未经授权禁止转载】et 获取更多精彩内容】【作者:唐霜】

但是,想象一下,你的系统中可能存在多个 【关注微信公众号:wwwtangshuangnet】本文版权归作者所有,未经授权不得转载。MyLib 的实例。那是不是意味着每一个【本文受版权保护】【原创内容,转载请注明出处】实例都要传入 url 进行加载?其实这个著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshuangnet】加载是没有必要的,因为同一个 scrip【作者:唐霜】【版权所有】唐霜 www.tangshuang.nett 脚本被加载之后,没有必要再去运行第二【原创内容,转载请注明出处】【版权所有】唐霜 www.tangshuang.net次,否则每次使用一个插件都要走加载+执行原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】两个步骤,严重拖长了执行时间。

本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。

我最终选择的方式是,你需要在全局注册需要原创内容,盗版必究。【关注微信公众号:wwwtangshuangnet】的插件,在实例上,使用这些插件即可。

【本文受版权保护】【未经授权禁止转载】
MyLib.registerPlugin(url)

const myLib = new MyLib()
myLib.usePlugin('some', options)

这样的好处是,同一个插件,一旦完成加载和【本文首发于唐霜的博客】【作者:唐霜】注册,那么就不需要再进行第二次加载。对于未经授权,禁止复制转载。原创内容,盗版必究。实例而言,可以在最快的时间,进入插件的执原创内容,盗版必究。未经授权,禁止复制转载。行阶段,从而拥有该插件提供的功能。

【未经授权禁止转载】未经授权,禁止复制转载。【本文首发于唐霜的博客】

插件的使用

但是,一定要记住,插件的加载和执行都是异原创内容,盗版必究。未经授权,禁止复制转载。步的。为什么执行也是异步的呢?比如上面的【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】代码中 myLib.usePlugin('some', options) 并不能在执行当前这一句代码的时候,就真【访问 www.tangshuang.net 获取更多精彩内容】【版权所有】唐霜 www.tangshuang.net正执行插件的定义函数,而是要等待插件加载未经授权,禁止复制转载。【未经授权禁止转载】完之后才会执行。为什么?因为我们在调用 【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】usePlugin 的时候,这个插件可能著作权归作者所有,禁止商业用途转载。【本文首发于唐霜的博客】还没有加载完呢。

【原创内容,转载请注明出处】【访问 www.tangshuang.n【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。et 获取更多精彩内容】

而且,我们的一些操作,必须等到插件加载执【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。行完之后才去运行,因此,我们需要将我们的【版权所有】唐霜 www.tangshuang.net著作权归作者所有,禁止商业用途转载。程序分段执行,例如:

【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。原创内容,盗版必究。【原创内容,转载请注明出处】
myLib.usePlugin('some', options, function() {
  this.some()
})

这段代码的意思是,当插件 some 已经准备好(加载完,并执行完)之后,执著作权归作者所有,禁止商业用途转载。【未经授权禁止转载】行回调函数中的内容。通过传入一个回调函数【转载请注明来源】【版权所有,侵权必究】,表示当插件真正加载完之后再运行这个函数【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】,而在这个函数内部,可以使用 this 【本文受版权保护】本文作者:唐霜,转载请注明出处。表示当前这个实例。但是,在一些情况下,我【转载请注明来源】著作权归作者所有,禁止商业用途转载。们可能需要同时等待多个插件加载完之后,才【转载请注明来源】【作者:唐霜】运行某个回调。我们可以这样子:

原创内容,盗版必究。【作者:唐霜】著作权归作者所有,禁止商业用途转载。【关注微信公众号:wwwtangshua著作权归作者所有,禁止商业用途转载。【访问 www.tangshuang.net 获取更多精彩内容】ngnet】
myLib.onPlugin(['a', 'b'], function() {
  this.a()
  this.b()
})

由于异步的加载和执行,给我们的插件使用带【版权所有,侵权必究】【未经授权禁止转载】来了麻烦,但是这就是代价,异步编程让我们未经授权,禁止复制转载。【转载请注明来源】没有那么方便的使用按照同步思维进行。不过【关注微信公众号:wwwtangshuangnet】【作者:唐霜】,为了更好的符合我们的编程习惯,我们可以【原创不易,请尊重版权】【本文首发于唐霜的博客】设计成 async/await 的模式。

【未经授权禁止转载】转载请注明出处:www.tangshua【版权所有,侵权必究】【本文受版权保护】ng.net
;(async function() {
  const myLib = new MyLib()

  await myLib.usePlugin('some', options) // 使用 await 等待插件加载和执行完
  await myLib.usePlugin('plugin2', options)

  myLib.some()
  myLib.b()
})()

我们将异步插件系统设计为返回一个 Pro【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】mise,这样就可以使用 async/a本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.netwait 进行编程,当然,这也不是必须的【本文受版权保护】著作权归作者所有,禁止商业用途转载。选择。

未经授权,禁止复制转载。著作权归作者所有,禁止商业用途转载。

插件的命名

一个问题是,我们让插件开发者命名自己的插【原创不易,请尊重版权】【本文受版权保护】件,还是让插件的使用者在注册插件时命名插本文作者:唐霜,转载请注明出处。【转载请注明来源】件?这里需要关心的一个问题是,如果两个插【未经授权禁止转载】【本文首发于唐霜的博客】件的开发者使用了相同的插件名,而插件的使原创内容,盗版必究。【本文首发于唐霜的博客】用者可能同时需要这两个插件。因此,我更倾【本文受版权保护】【作者:唐霜】向让插件的使用者给插件命名。

【版权所有,侵权必究】原创内容,盗版必究。

然而,这也会带来一些副作用。实际上,我们【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.net使用异步插件的目的,是希望减小项目本身的【版权所有】唐霜 www.tangshuang.net【本文受版权保护】打包代码,加快代码初始化速度。但是,倘若【本文受版权保护】【原创不易,请尊重版权】我们给了插件使用者更多选择,那么势必我们【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.net的目的会遭受调整,项目的代码可能会由于这未经授权,禁止复制转载。【作者:唐霜】些逻辑,多出一些原本可能不必要的逻辑代码【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.net。这是取舍两难的话题。

本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshua【版权所有,侵权必究】转载请注明出处:www.tangshuang.netng.net【原创不易,请尊重版权】

为了和 AMD 契合,我最终选择了让插件【作者:唐霜】【版权所有】唐霜 www.tangshuang.net的开发者来定义自己的插件名字。

【版权所有,侵权必究】【版权所有】唐霜 www.tangshu未经授权,禁止复制转载。本文作者:唐霜,转载请注明出处。ang.net【未经授权禁止转载】【转载请注明来源】

让插件开发者定义插件的名字,这样,一个插本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net件拥有什么名字,就是固定的了。这样的好处原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】非常多,除了可能的命名冲突之外,全是好处【本文受版权保护】著作权归作者所有,禁止商业用途转载。。当然,作为插件系统,最好对该命名有一个本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】规范要求,例如要求插件名有一个 scop未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.nete,这样就可以通过 scope 来区分不本文作者:唐霜,转载请注明出处。【转载请注明来源】同开发者提供的不同插件。

未经授权,禁止复制转载。【作者:唐霜】

而且,这样的更大的好处在于,对依赖的确定著作权归作者所有,禁止商业用途转载。【访问 www.tangshuang.net 获取更多精彩内容】更加有利。插件的开发者可以非常明确,自己【版权所有,侵权必究】【本文首发于唐霜的博客】的插件是依赖哪一个插件的,这个插件在当前本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】插件中运行起来没有任何问题。但假如插件名转载请注明出处:www.tangshuang.net【关注微信公众号:wwwtangshuangnet】不是有插件开发者自己规定,而是插件使用者【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net在自己系统中动态规定,那么就会导致当前这未经授权,禁止复制转载。【版权所有,侵权必究】个插件完全不知道自己该使用什么样的依赖,【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】以及是否正确加载了依赖的插件。

【版权所有】唐霜 www.tangshu【原创不易,请尊重版权】【原创不易,请尊重版权】ang.net【关注微信公众号:wwwtangshua原创内容,盗版必究。【本文首发于唐霜的博客】ngnet】

而且,这样的好处还在于,可以更大限度的节本文作者:唐霜,转载请注明出处。【作者:唐霜】省项目初始化脚本文件的大小,让插件可以容原创内容,盗版必究。未经授权,禁止复制转载。纳更多代码,从而达到我们异步加载的目的。

【关注微信公众号:wwwtangshua【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.netngnet】【关注微信公众号:wwwtangshua【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】ngnet】著作权归作者所有,禁止商业用途转载。

插件预处理

在一些使用场景中,插件还会尝试在加载之前【未经授权禁止转载】【原创不易,请尊重版权】收集一些信息,加载结束的时候利用这些信息【原创内容,转载请注明出处】【转载请注明来源】进行下一步操作。因此,插件的注册,还需要【原创内容,转载请注明出处】【作者:唐霜】安装一定的方式进行重新设计。在同步任务中【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。,插件需要收集数据,如下:

原创内容,盗版必究。【转载请注明来源】【未经授权禁止转载】转载请注明出处:www.tangshua未经授权,禁止复制转载。【原创内容,转载请注明出处】ng.net
MyLib.registerPlugin({
  url, // 插件的 url
  options, // 插件的基础参数,后续启用插件时所传入的参数会和这个 options 进行合并
  onReady, // 插件注册完成之后立即运行的函数
  onInit, // 实例化时运行的函数,支持 this
  onLoad, // 插件加载完成时运行的函数
  onPlugin, // 插件函数被调用后运行的函数,支持 this
  onDestroy, // 实例销毁时运行的函数
})

通过这样的配置,可以更全面的控制插件。在【关注微信公众号:wwwtangshuangnet】【作者:唐霜】 onReady 和 onInit 中,【未经授权禁止转载】【原创内容,转载请注明出处】可以进行一些数据收集。对于插件的作者而言【作者:唐霜】【作者:唐霜】,不需要再暴露 url 给插件的使用者,【本文首发于唐霜的博客】【关注微信公众号:wwwtangshuangnet】而是直接提供一个封装好的 js 模块给使本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。用者调用。例如:

【版权所有】唐霜 www.tangshu转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。ang.net本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。
import SomePlugin from './some-plugin.js'
MyLib.registerPlugin(SomePlugin)

而这个 some-plugin.js 的【本文受版权保护】【未经授权禁止转载】实际内容可能是:

转载请注明出处:www.tangshua【本文受版权保护】未经授权,禁止复制转载。ng.net【本文首发于唐霜的博客】本文版权归作者所有,未经授权不得转载。
let count = 0
const SomePlugin = {
  url,
  onInit() {
    this.count = count
    count ++
  }
}

这个插件的作用,就是在实例上面添加了一个【转载请注明来源】原创内容,盗版必究。 count 属性,而这个 count 【本文受版权保护】【版权所有,侵权必究】属性,正好在我们的插件的定义函数中使用。【作者:唐霜】原创内容,盗版必究。总之,这种方式帮助插件的开发者更好的做一本文作者:唐霜,转载请注明出处。【本文首发于唐霜的博客】些处理。

【原创不易,请尊重版权】【本文首发于唐霜的博客】

插件注册的实现

这一节,我们用实际的代码来实现异步插件的转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。加载。前文实际上已经分析过了,我们可以使【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。用 script 标签来做异步加载,只需原创内容,盗版必究。【原创不易,请尊重版权】要开启 script.async 属性即本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】可,这也是为什么我们不使用 xhr + 【转载请注明来源】【未经授权禁止转载】script.text 的原因,因为使用【本文受版权保护】原创内容,盗版必究。 script.text 是同步执行的,【关注微信公众号:wwwtangshuangnet】未经授权,禁止复制转载。也就是说,插件的内容会在当前线程环境中立未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net即执行。因此,我们使用 src 来实现更【版权所有,侵权必究】【转载请注明来源】好。

著作权归作者所有,禁止商业用途转载。【版权所有,侵权必究】【原创不易,请尊重版权】本文版权归作者所有,未经授权不得转载。

让我们回到真实的场景中,对于用户而言,可【原创内容,转载请注明出处】【转载请注明来源】能是这样使用插件:

原创内容,盗版必究。【版权所有,侵权必究】
<!DOCTYPE html>
<head>
  <script src="http://cdn.xxx.com/my-lib.min.js"></script>
  <script>
    MyLib.registerPlugin('http://cdn.xxx.com/plugins/@tomy/touch.min.js')
    MyLib.registerPlugin('http://cdn.xxx.com/plugins/@pony/touch.min.js') // 虽然插件名相同,但是 scope 不同
  </script>
</head>

我们来实现这个 registerPlug【版权所有,侵权必究】【转载请注明来源】in 方法:

转载请注明出处:www.tangshua原创内容,盗版必究。【转载请注明来源】ng.net【未经授权禁止转载】【访问 www.tangshuang.n【原创内容,转载请注明出处】【原创内容,转载请注明出处】et 获取更多精彩内容】本文作者:唐霜,转载请注明出处。
MyLib._instances = []
MyLib._plugins = []
MyLib.registerPlugin = function(def) {
  // 如果是直接传入 url
  if (typeof def === 'string') {
    def = { url: def }
  }
  
  const { url, required, options, onReady, onInit, onPlugin, onLoaded, onDestroy } = def
  const id = 'plugin' + Math.random()
  
  const plugin = {
    id,
    required,
    options,
    onInit,
    onPlugin,
    onDestroy
  }

  if (typeof onReady === 'function') {
    onReady()
  }

  const script = document.createElement('script')
  script.async = true
  script.defer = true
  script.id = id
  script.src = url
  script.onload = () => {
    if (typeof onLoad === 'function') {
      onLoad()
    }

    // 决定依赖关系,并按依赖关系排序
    this._plugins.sort((a, b) => {
      const aName = a.name
      const aDeps = a.deps
      const bName = b.name
      const bDeps = b.deps
      
      if (aDeps.includes(bName)) {
        return -1
      }
      else if (bDeps.includes(aName)) {
        return 1
      }
      else {
        return 0
      }
    })

    MyLib._instances.forEach((mylib) => {
      // TODO: 当一个插件加载之后,要去检查所有实例是否已经使用了插件,如果使用了插件,就直接启用插件
      mylib._runPlugins()
    })
  }

  this._plugins.push(plugin)
}

对于插件的开发者而言,要在异步脚本中使用著作权归作者所有,禁止商业用途转载。【作者:唐霜】 MyLib.defineAsyncPl【原创内容,转载请注明出处】【原创不易,请尊重版权】ugin 来定义插件,他可能这么写:

著作权归作者所有,禁止商业用途转载。【原创内容,转载请注明出处】本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。
MyLib.defineAsyncPlugin('@someone/count', [], function() {
  console.log(this.count)
})

那么,这个 defineAsyncPlu本文作者:唐霜,转载请注明出处。【本文受版权保护】gin 怎么实现呢?

转载请注明出处:www.tangshua原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.netng.net【未经授权禁止转载】
MyLib.defineAsyncPlugin = function(name, deps, define) {
  const id = document.currentScript.id
  const plugin = this._plugins.find(item => item.id === id)
  if (!plugin) {
    return
  }

  Object.assign(plugin, {
    name,
    deps,
    define,
  })
}

这样,当开发者在自己的项目中注册了这个插【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。件,就可以正常的加载到项目系统中了。而且本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。,我们还有一个 runPlugins 的转载请注明出处:www.tangshuang.net【作者:唐霜】方法,这个方法会去检查所有实例上应该被执本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net行的插件。当然,如果这些插件的依赖没有执【转载请注明来源】【关注微信公众号:wwwtangshuangnet】行完,那么它们自己本身也不会被执行。

【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。

插件启用的实现

我们已经完成了插件的注册,那么,接下来,原创内容,盗版必究。【未经授权禁止转载】我们在实例中启用这个插件,会遇到哪些需要本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net解决的问题呢?启用插件因为是在实例上进行【未经授权禁止转载】【本文受版权保护】,因此它有两种选择,一种是为实例的初始化【本文首发于唐霜的博客】【版权所有,侵权必究】参数设计一个配置选项,另一种是使用方法发【转载请注明来源】【转载请注明来源】起命令。而在使用配置参数时,实际上也可以【访问 www.tangshuang.net 获取更多精彩内容】【作者:唐霜】是调用发起命令完成的。因此,我们需要定义著作权归作者所有,禁止商业用途转载。本文作者:唐霜,转载请注明出处。发起调用插件的方法:

转载请注明出处:www.tangshua原创内容,盗版必究。本文作者:唐霜,转载请注明出处。ng.net【原创内容,转载请注明出处】【本文首发于唐霜的博客】【作者:唐霜】
class MyLib {
  _pluginQueue = []
  _pluginModules = {}

  constructor(options = {}) {
    const { plugins } = options
    if (plugins) {
      const names = Object.keys(plugins)
      names.forEach((name) => {
        const options = plugins[name]
        this.usePlugin(name, options)
      })
    }

    MyLib._instances.push(this)
  }
  
  usePlugin(name, options = {}) {
    // 实际上,代码非常非常简单,就是将该插件的启用信息,加入到一个队列中
    this._pluginQueue.push({ name, options })
    this._runPlugins()
    return this
  }

  onPlugin(names, fn) {
    if (typeof names === 'string') {
      names = [names]
    }

    this._pluginQueue.push({ names, fn })
    this._emitPlugins()
  }

  // 这个方法用于检查全部的插件是否已经被运行,如果没有被运行,那么根据其依赖,执行它
  _runPlugins() {
    const queue = this._pluginQueue
    const modules = this._pluginModules
    const plugins = MyLib._plugins
    
    if (!Object.keys(queue).length) {
      return
    }

    plugins.forEach((plugin) => {
      const { name, deps, define, onPlugin, options: basicOptions } = plugin
      if (name in modules) {
        return
      }

      const request = queue.find(item => item.name === name)
      if (!request) {
        return
      }

      // 检查所有依赖是否已经加载好了
      if (deps.find(item => !(item in modules)) {
        return
      }

      // 记录真正的 options
      const requestOptions = request.options
      const options = { ...basicOptions, ...requestOptions }

      const injects = deps.map(item => item === '$options' ? options : modules[item])
      const module = define.apply(this, injects)
      modules[name] = module
      
      if (typeof onPlugin === 'function') {
        onPlugin.call(this, module)
      }

      this._emitPlugins()
    })
  }

  _emitPlugins() {
    const queue = this._pluginQueue
    const modules = this._pluginModules
    queue.forEach((item) => {
      if (item.done || !item.names) {
        return
      }
      if (item.names.find(name => !modules[name])) {
        return
      }
      item.fn.call(this)
      item.done = true
    })
  }
}

这样,插件的使用者,就可以使用我们提供的著作权归作者所有,禁止商业用途转载。原创内容,盗版必究。这些方法,启动插件,使得当前这个实例,拥【版权所有】唐霜 www.tangshuang.net【版权所有】唐霜 www.tangshuang.net有插件提供的功能。

【本文首发于唐霜的博客】未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。
<script>
const myLib = new MyLib()
myLib.usePlugin('@tomy/touch')
  .usePlugin('@pony/touch', { debounce: 2000 })
  .onPlugin(['@tomy/touch', '@pony/touch'], function() {
    this.on('touch', e => console.log(e))
    this.touch()
  })
</script>

我们没有实现 async/await 方【未经授权禁止转载】转载请注明出处:www.tangshuang.net案,因为这样又会多出一些代码,我们尽可能【原创内容,转载请注明出处】【未经授权禁止转载】的减少代码量。

【本文受版权保护】【转载请注明来源】【转载请注明来源】原创内容,盗版必究。

结束语

本文分析并实现了一个异步插件系统的骨架方原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】案,这个异步插件系统立足于“类-实例”的【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。结构方式,而非单例模式,因此设计上会比单【访问 www.tangshuang.net 获取更多精彩内容】【版权所有】唐霜 www.tangshuang.net例模式的系统更麻烦一些。回到文章开头,我转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。们这样做的目的,是为了将一些插件拆分出来【关注微信公众号:wwwtangshuangnet】原创内容,盗版必究。,根据实际的需要来定制需要加载的脚本,从【原创内容,转载请注明出处】著作权归作者所有,禁止商业用途转载。而减少页面初始脚本的大小,以及初始执行所【版权所有】唐霜 www.tangshuang.net【原创内容,转载请注明出处】占用的时间。但这也不可避免的带来了编程上【版权所有,侵权必究】【未经授权禁止转载】的一些麻烦。有舍才有得,这不一定是最好的【未经授权禁止转载】【关注微信公众号:wwwtangshuangnet】方案,更不是放之四海而皆准的方案,只有在转载请注明出处:www.tangshuang.net【原创内容,转载请注明出处】你的项目需要通过这种方式加载更有利的情况【原创内容,转载请注明出处】本文版权归作者所有,未经授权不得转载。下,才使用这种方案,才是正确的选择。

原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。

2020-01-12 4683 ,

为价值买单,打赏一杯咖啡

本文价值46.83RMB