JavaScript 模块的循环加载

483 查看

“循环加载”(circular dependency)指的是,a脚本的执行依赖b脚本,而b脚本的执行又依赖a脚本。

通常,”循环加载”表示存在强耦合,如果处理不好,还可能导致递归加载,使得程序无法执行,因此应该避免出现。

但是实际上,这是很难避免的,尤其是依赖关系复杂的大项目,很容易出现a依赖b,b依赖cc又依赖a这样的情况。这意味着,模块加载机制必须考虑”循环加载”的情况。

本文介绍JavaScript语言如何处理”循环加载”。目前,最常见的两种模块格式CommonJS和ES6,处理方法是不一样的,返回的结果也不一样。

CommonJS模块

CommonJS模块的重要特性是加载时执行,即脚本代码在require的时候,就会全部执行。因此,CommonJS规定,一旦发现某个模块被”循环加载”,就立即停止加载,只输出已经执行的部分。

让我们来看,官方文档里面的例子。脚本文件a.js代码如下。

上面代码之中,a.js脚本先输出一个done变量,然后加载另一个脚本文件b.js。注意,此时a.js代码就停在这里,等待b.js执行完毕,再往下执行。

再看b.js的代码。

上面代码之中,b.js执行到第二行,就会去加载a.js,这时,就发生了”循环加载”。为了避免无穷递归,执行引擎不会去再次执行a.js,而是只返回已经执行的部分。

a.js已经执行的部分,只有一行。

因此,对于b.js来说,它从a.js只输入一个变量done,值为false

然后,b.js接着往下执行,等到全部执行完毕,再把执行权交还给a.js。于是,a.js接着往下执行,直到执行完毕。我们写一个脚本main.js,验证这个过程。

执行main.js,运行结果如下。

上面的代码证明了两件事。一是,在b.js之中,a.js没有执行完毕,只执行了第一行。二是,main.js执行到第二行时,不会再次执行b.js,而是输出缓存的b.js的执行结果,即它的第四行。

ES6模块

ES6模块的运行机制与CommonJS不一样,它遇到模块加载命令import时,不会去执行模块,而是只生成一个引用。等到真的需要用到时,再到模块里面去取值。

因此,ES6模块是动态引用,不存在缓存值的问题,而且模块里面的变量,绑定其所在的模块。请看下面的例子。