"Circular dependency" means that the execution of script a depends on script b, and the execution of script b depends on script 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.
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.
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.
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.
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.
Execute main.js and the results are as follows.
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.
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.
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.
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).
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.
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.
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.
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.
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.
(over) |
Text/E-commerce consultant Lao Lu First summarize...
We often hear about cases of hit activities and p...
Preface: Some of the friends who read this articl...
Poor conversion - being scolded by the boss - low...
[[162981]] On the eve of the opening of MWC 2016 ...
Today let’s talk about the 10-yuan operational th...
Today I will share a foreign online earning proje...
Only by improving the conversion rate at each ste...
Brand building is a long process. Although it is ...
My name is Lao Huang. I am an operator. In fact, ...
"The Big Dipper points to the southeast, and...
According to 9to5 Mac, more and more iPhone users...
Information flow promotion often encounters vario...
This issue provides you with ideas for optimizing...
Tik Tok has been popular since early 2018 and is ...