Ruan Yifeng: Cyclic loading of JavaScript modules

Ruan Yifeng: Cyclic loading of JavaScript modules

"Circular dependency" means that the execution of script a depends on script b, and the execution of script b depends on script a.

  1. // a.js  
  2. var b = require( 'b' );
  3.  
  4. // b.js  
  5. var a = require( 'a' );

Generally, "cyclic loading" indicates strong coupling, and if not handled properly, it can lead to recursive loading, making the program unexecutable, so it should be avoided.

[[154576]]

But in reality, this is difficult to avoid, especially in large projects with complex dependencies, where it is easy for a to depend on b, b to depend on c, and c to depend on a. This means that the module loading mechanism must consider the situation of "cyclic loading".

This article introduces how JavaScript handles "cyclic loading". Currently, the two most common module formats, CommonJS and ES6, have different processing methods and return different results.

CommonJS modules

The important feature of CommonJS modules is that they are executed when loaded, that is, the script code will be executed in full when it is required. Therefore, CommonJS stipulates that once a module is found to be "loaded in a loop", the loading will be stopped immediately and only the executed part will be output.

Let's take a look at the example in the official documentation. The code of the script file a.js is as follows.

  1. exports.done = false ;
  2. var b = require( './b.js' );
  3. console.log( 'In a.js, b.done = %j' , b.done);
  4. exports.done = true ;
  5. console.log( 'a.js completed execution' );

In the above code, the a.js script first outputs a done variable, and then loads another script file b.js. Note that the a.js code stops here and waits for b.js to finish executing before continuing.

Let’s look at the code of b.js again.

  1. exports.done = false ;
  2. var a = require( './a.js' );
  3. console.log( 'In b.js, a.done = %j' , a.done);
  4. exports.done = true ;
  5. console.log( 'b.js completed execution' );

In the above code, when b.js executes to the second line, it will load a.js. At this time, "loop loading" occurs. In order to avoid infinite recursion, the execution engine will not execute a.js again, but only return the executed part.

The executed part of a.js is only one line.

  1. exports.done = false ;

Therefore, for b.js, it only inputs one variable done from a.js, and the value is false.

Then, b.js continues to execute until all execution is completed, and then returns the execution right to a.js. So, a.js continues to execute until the execution is completed. Let's write a script main.js to verify this process.

  1. var a = require( './a.js' );
  2. var b = require( './b.js' );
  3. console.log( 'In main.js, a.done=%j, b.done=%j' , a.done, b.done);

Execute main.js and the results are as follows.

  1. $ node main.js
  2.  
  3. In b.js, a.done = false  
  4. b.js is executed
  5. In a.js, b.done = true  
  6. a.js is executed
  7. In main.js, a.done = true , b.done = true  

The above code proves two things. First, a.js is not completely executed in b.js, only the first line is executed. Second, when main.js executes to the second line, it will not execute b.js again, but output the cached execution result of b.js, that is, its fourth line.

  1. exports.done = true ;

ES6 modules

The operating mechanism of ES6 modules is different from that of CommonJS. When it encounters the module loading command import, it will not execute the module, but only generate a reference. When it is really needed, it will then get the value from the module.

Therefore, ES6 modules are dynamically referenced, there is no problem of cached values, and the variables in the module are bound to the module it is in. Please see the example below.

  1. // m1.js  
  2. export var foo = 'bar' ;
  3. setTimeout(() => foo = 'baz' , 500 );
  4.  
  5. // m2.js  
  6. import {foo} from './m1.js' ;
  7. console.log(foo);
  8. setTimeout(() => console.log(foo), 500 );

In the above code, the variable foo of m1.js is equal to bar when it is first loaded, and becomes equal to baz after 500 milliseconds.

Let's see if m2.js can read this change correctly.

  1. $ babel-node m2.js
  2.  
  3. bar
  4. baz

The above code shows that ES6 modules do not cache the running results, but dynamically get the values ​​of the loaded modules, and the variables are always bound to the module in which they are located.

This makes ES6's handling of "cyclic loading" fundamentally different from CommonJS. ES6 does not check whether "cyclic loading" has occurred at all, but only generates a reference to the loaded module. It is up to the developer to ensure that the value can be obtained when the value is actually obtained.

See the following example (taken from Dr. Axel Rauschmayer's Exploring ES6).

  1. // a.js  
  2. import {bar} from './b.js' ;
  3. export function foo() {
  4. bar();
  5. console.log( 'Execution completed' );
  6. }
  7. foo();
  8.  
  9. // b.js  
  10. import {foo} from './a.js' ;
  11. export function bar() {
  12. if (Math.random() > 0.5 ) {
  13. foo();
  14. }
  15. }

According to the CommonJS specification, the above code cannot be executed. a first loads b, and then b loads a. At this time, a has not yet executed any results, so the output result is null. That is, for b.js, the value of the variable foo is equal to null, and the subsequent foo() will report an error.

However, ES6 can execute the above code.

  1. $ babel-node a.js
  2.  
  3. Execution completed

The reason why a.js can be executed is that the variables loaded by ES6 are all dynamically referenced to the module they are in. As long as the reference exists, the code can be executed.

Let's look at an example given by the ES6 module loader SystemJS.

  1. // even.js  
  2. import { odd } from './odd'  
  3. export var counter = 0 ;
  4. export function even(n) {
  5. counter++;
  6. return n == 0 || odd(n - 1 );
  7. }
  8.  
  9. // odd.js  
  10. import { even } from './even' ;
  11. export function odd(n) {
  12. return n != 0 && even(n - 1 );
  13. }

In the above code, the function foo in even.js has a parameter n. As long as it is not equal to 0, it will subtract 1 and pass it into the loaded odd(). odd.js will also do a similar operation.

Running the above code, the results are as follows.

  1. $ babel-node
  2. > import * as m from './even.js' ;
  3. > m.even( 10 );
  4. true  
  5. > m.counter
  6. 6  
  7. > m.even( 20 )
  8. true  
  9. > m.counter
  10. 17  

In the above code, when the parameter n changes from 10 to 0, foo() will be executed 6 times in total, so the variable counter is equal to 6. When even() is called for the second time, the parameter n changes from 20 to 0, and foo() will be executed 11 times in total, plus the previous 6 times, so the variable counter is equal to 17.

If this example is rewritten as CommonJS, it will not be executable at all and will report an error.

  1. // even.js  
  2. var odd = require( './odd' );
  3. var counter = 0 ;
  4. exports.counter = counter;
  5. exports.even = function(n) {
  6. counter++;
  7. return n == 0 || odd(n - 1 );
  8. }
  9.  
  10. // odd.js  
  11. var even = require( './even' );
  12. exports.odd = function(n) {
  13. return n != 0 && even(n - 1 );
  14. }

In the above code, even.js loads odd.js, and odd.js loads even.js, forming a "loop loading". At this time, the execution engine will output the part that even.js has executed (there is no result), so in odd.js, the variable even is equal to null, and an error will be reported when even(n-1) is called later.

  1. $ node
  2. > var m = require( './even' );
  3. > m.even( 10 )
  4. TypeError: odd is not a function

(over)

<<:  Multi-dimensional exploration of HTML5 technology to create the best application experience and practical sharing

>>:  WOT lecturer, Taobao mobile technical expert Chen Wu: The big data collection system behind Taobao mobile's billions of UVs

Recommend

Absolutely useful information! Nine abnormal ways of operating Taobao and Tmall

Text/E-commerce consultant Lao Lu First summarize...

Tips for creating hit events!

We often hear about cases of hit activities and p...

MWC technology giants lead the VR craze and the global melee will begin

[[162981]] On the eve of the opening of MWC 2016 ...

Let's talk about the operating thinking and skills of 10 yuan

Today let’s talk about the 10-yuan operational th...

APP promotion: conversion rate increased by 1750%! I did it with these tips

Only by improving the conversion rate at each ste...

What is the difference between a good operator and an ordinary junior operator?

My name is Lao Huang. I am an operator. In fact, ...

Some Apple iPhone 13/12 iOS 15 devices damage car Bluetooth hands-free system

According to 9to5 Mac, more and more iPhone users...

How to optimize e-commerce promotion costs if they are too high?

Information flow promotion often encounters vario...

k12 advertising creative optimization guide!

This issue provides you with ideas for optimizing...

2019 Tik Tok Product Analysis Report!

Tik Tok has been popular since early 2018 and is ...