Module this thing really should take care of a reason, some knowledge points summed up later need not see, refueling duck! Wrote me for 2 days -_-

This article answers these key questions:

1. What is load at compile time and what is load at run time 2. CMD commonJS module export/module.exports/commonjs/commonjs/commonjs/commonjs/commonjs/commonjs/commonjs/commonjs/commonjs

What is a module? As well as the role

The internal data and implementation of a module are private, exposing only interfaces (methods) to the outside to communicate with other modules

Functions: 1. Avoid naming conflicts (reduce namespace pollution) 2. Flexible architecture, focus separation, convenient module combination, decomposition 3. 4. High reusability and maintainability

2. Compilation principle

In the flow of a traditional compiled language, a piece of source code in a program goes through three steps, collectively called “compilation,” before it is executed.

2.1 Word Segmentation/Lexical Analysis (Tokenizing/Lexing)

This process breaks down a string of characters into blocks of code that make sense (to the programming language), called lexical units (tokens). For example, consider the program var a = 2; . This program is usually broken down into the following lexical units: var, a, =, 2; . Whether Spaces are considered lexical units depends on whether they make sense in the language

2.2 Parsing/Parsing

This process converts a stream of lexical units (arrays) into a hierarchical nested tree of elements that represents the syntactic structure of the program. This Tree is called the Abstract Syntax Tree (AST). var a = 2; The abstract syntax tree may have a top-level node called VariableDeclaration, followed by a child node called Identifier (whose value is a), and a child node called AssignmentExpression. The AssignmentExpression node has a child called NumericLiteral (whose value is 2)

2.3 Code Generation

The process of turning an AST into executable code is called code generation. This process depends on language, target platform, and so on. Regardless of the details, the short answer is that there is some way to set var a = 2; The AST is converted to a set of machine instructions that create a variable called A (including allocating memory, etc.) and store a value in A.

2.4 Differences between JS and other languages in compilation

JavaScript engines are much more complex than compilers for languages that compile in three steps. For example, there are specific steps in the parsing and code generation phases to optimize runtime performance, including for redundant elements.

With JavaScript, most of the time compilation happens a few microseconds (or less!) before the code executes. Time. Behind the scope we will discuss, JavaScript engines use various methods (such as JIT, which can delay compilation or even recompile) to ensure optimal performance.

Any snippet of JavaScript code is compiled before execution (usually just before execution). Therefore, the JavaScript compiler first evaluates var a = 2; The program compiles, is ready to execute, and usually executes immediately

Summary above:

Js code will go through the compilation stage before execution (word segmentation/lexical analysis -> parsing/parsing -> code generation), why compile before execution? Because JS language is a high-level programming language, easy to read and write, and the target language is machine language, that is, binary code, can be directly recognized by the computer.

3. AMD (the require. Js) – the browser

Prior to ES6, there were several module loading schemes developed by the community, the most important being CommonJS and AMD. The former is used for servers and the latter for browsers.

3.1 Use of AMD

The directory structure

require.html

<! DOCTYPEhtml>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="Width = device - width, initial - scale = 1.0">
  <title>Document</title>
</head>
<body>
  <script src="./require.js" data-main="./main.js" defer async="true" ></script> 
  <! -- import require.js package, import main.js -->
</body>
</html>
Copy the code

main.js

console.log(require.'Main.js begins execution')
require.config({baseUrl: "./common/".// Extract the public path
 paths: {"module1": "module1"."module2": "module2",}})require(['module1'.'module2'].function (module1, module2){// some code here
  console.log(module1.add(5.6), 'module1')
  console.log(module2.addUnit(5.6),'module2')});Copy the code

module1.js

console.log('module1')
define(function (){const add = (x,y) = >x+yreturn{the add}; });Copy the code

module2.js

console.log('module2')
define(['./module1'].function (module){constAddUnit =(x,y) = > `The ${module.add(x,y)}Yuan ` // Module2 depends on module1
  return{addUnit}; });Copy the code

Require. js loading module, using AMD specifications. That is, modules must be written as specified by AMD.

Specifically, modules must be defined with a specific define() function. If a module does not depend on other modules, it can be defined directly in the define() function.

AMD calls it "run time loading" because only the object is available at run time, making it completely impossible to "static optimize" at compile time.

3.2 What problems did AMD solve

1. Before AMD, we used the SRC tag of script to introduce a large number of JS. The larger the project, the more JS will be introduced, which is very ugly. Module2 depends on module1, so require([‘module1’, When ‘module2’], we should load module1 first and then module2 so that it can be loaded in sequence 3. Defer async=”true” to allow js to load asynchronously and avoid page script loading multiple JS with no response 4. 5. Require () implement load on demand, there is a feature only one page (a single page of a component) to introduce a library, you do not need to load on demand is equivalent to every page to load the library

4.CommonJS – Server (Node.js)

The characteristics of the 4.1 CommonJS

In the 2.node. js module system, each file is treated as an independent module. Modules include CommonJS core variables: exports, module.exports, require 3. The core of block is export and import. Node exports the contents of modules through exports and module.exports, and imports the contents of other modules (custom modules, system modules and third-party library modules) through require function

4.2CommonJS Basic usage

File directory

To run the node file, simply CD to the node+ file

node1.js

const res = require('./node2.js');
console.log(res.obj);
Copy the code

node2.js

exports.obj = {
  text1: 'Welcome to the public account'.text2: '[The programmer loves to talk]',}Copy the code

Module. Exports + require() exports + require(

node1.js

const res = require('./node2.js');
console.log(res(5))
Copy the code

node2.js

module.exports = (a) = > a + 1
Copy the code

4.3 What is the difference between exports and Module. exports

For this piece, many people write so much praise, is really afraid! So my understanding is as follows, there are mistakes please correct!!

A reference to module.exports, whose input is shorter, allows a shortcut so that module.exports. It can be written more succinctly as exports. F =… . Note, however, that as with any variable, if you assign a new value to exports, it is no longer bound to module.exports

module.exports.hello = true; // Exports from module require = {hello: false}; // Not exported, only available in modules

Exports: module. Exports: module. Exports: module.

    const module = {
      exports: {}}const exports = module.exports

    exports.name = 'Welcome to the program yuan love to talk' // exports is assigned, but module.exports is still called
    console.log(module.exports, Example '1');  // In this case, yes
Copy the code

    const module = {
      exports: {}}const exports = module.exports

    exports = 'Welcome to the program yuan love to talk' // exports is assigned, but module.exports is still called
    console.log(module.exports, 'example 2');  // In this case,exports are no longer referenced, so no
Copy the code

So to sum up:

Module. exports can be used in any scenario, because the last call to module.exports is used when it is not reassigned, i.e. Module. exports= {/ / expor} Ts is the equivalent of a cut off reference, as long as you remember the difference, most of the scenes will still be able to be abbreviated using exports for simplicity

4.4 CommonJs features

1. All code runs in module scope and does not pollute global scope; 2. The module is loaded synchronously, that is, the following operations can be performed only after the loading is completed; The output of 3.CommonJS is a copy of the value (that is, the value returned by require is a copy of the value being output, and changes within the module do not affect this value).

See the examples:

node0.js

const res = require('./node1.js');
console.log(res(4.5))
console.log('node0')
Copy the code

node1.js


const res = require('./node2.js');
module. Exports =(x,y) = > `${res(x, y)}Yuan `
console.log('node1')
Copy the code

node2.js

module.exports = (x,y) = > x + y
console.log('node2')
Copy the code

Node0 has RES,node1 has RES, and node1 has RES, which does not affect the global scope diagram. This means that all code runs in the module scope and does not pollute the global scope diagram print

Third, CommonJS output is a copy of the value, that is, you import modules with require(), but how you change them in the latest module does not affect modules you already require(). See 6.1

5. ES6 Module –(Browser/Server)

5.1 How to Use it

The export to use

index.html

<! DOCTYPEhtml>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="Width = device - width, initial - scale = 1.0">
  <title>Document</title>
</head>
<body>
  <script type="module">
    import { text1, text2 } from './module1.js'
    console.log(text1, The '-', text2());
    
    // import * as module from './module1.js' can also be imported entirely
    // console.log(module.text1, '-', module.text2());
  </script>
</body>
</html>
Copy the code

module1.js

export const text1 = 'Welcome to the public account';
export const text2 = () = > '[The programmer loves to talk]';
Copy the code

The important thing to note here is that running directly will cause cross-domain problems. For example, SRC of script tags does not have cross-domain problems, so we need to ensure same-origin, install live server in vscode, and run

2.export default

module2.js

export default function test() {
  console.log('Welcome to pay attention to my public number -[program yuan love to chat]');
}
Copy the code

index.html

 <script type="module">
    import test from './module2.js'
    test()
  </script>
Copy the code

5.2 Differences between Export and Export default

1. When importing the command, the user needs to know the name of the variable or function to be loaded; otherwise, it cannot be loaded. Use the export default command to specify the default output 2 for the module. A page can have multiple exports but only one Export default

6.CommonJS,AMD,ES6 Module differences

CommonJS module`require()`Synchronous loading module, ES6 module`import`Commands are loaded asynchronously, with a separate module-dependent parsing phase. AMD asynchronously loads CommonJS modules with a copy of a value, ES6 modules with a reference to a value. CommonJS module, AMD is run time load, ES6 module is compile time output interface.Copy the code

The 6.1 CommonJS module outputs a copy of a value, the ES6 module outputs a reference to a value.

In fact there are a lot of people don’t know, I, too, but after some validation, I understand is copying a value, if it is the reference data types, we copy is actually his memory address (pointer), but are essentially called worth copy, rather than a pointer to copy, such as string pointer, copy duplicate values, just see value represents What is it.

See the examples:

node1.js

var mod = require('./node2');
console.log(mod.counter);  / / 3
mod.incCounter();
console.log(mod.counter); / / 3
Copy the code

node2.js

var counter = 3
function incCounter() {
  counter++;
}
module.exports = {
  counter: counter,
  incCounter: incCounter,
};
Copy the code

The value I’ve introduced above is a string, and it doesn’t work if I do anything to a string after I’ve assigned it, so the value here is a string

Is equal to:

let a = 1
let b = a
++a
console.log(b); / / 1
Copy the code

What if it’s an object? node1.js

var mod = require('./node2');

console.log(mod.counter);  // { count: 3 }
mod.incCounter();
console.log(mod.counter); // { count: 4 }
Copy the code

node2.js

var counter = {
  count: 3
}
function incCounter() {
  counter.count++;
}
module.exports = {
  counter: counter,
  incCounter: incCounter,
};
Copy the code

The value I introduced above is an object (memory address), which is valid after I assign it to the object, and the value here is a memory address

In ES5module, module1.js

export let counter = 3;
export function incCounter() {
  counter++;
}
Copy the code

index.html

  <script type="module">
    import { counter, incCounter } from './module1.js';
    console.log(counter); / / 3
    incCounter();
    console.log(counter); / / 4
  </script>
Copy the code

When the JS engine statically analyzes a script, it generates a read-only reference to the module load command import. When the script is actually executed, it will be evaluated in the loaded module based on the read-only reference. In other words, the IMPORT of ES6 is a bit like the “symbolic link” of Unix systems, where the original value changes and the import load value changes with it. Therefore, ES6 modules are referenced dynamically and do not cache values. Variables in modules are bound to the module in which they are located.

CommonJS outputs a value that is an operation on the corresponding value. Es6 module outputs a map object with a key as a path and a value as a value (regardless of whether the value is a string or an object) as a reference, and then points to it in the heap The same thing.

See: What is your import compiled into Webpack?

I have also tried to view the source code compiled by Webpack, and feel that I have not been able to parse the source code ~ but this picture can illustrate something.

For the import of this feature, I would need a bug, is my pot, of course, I was in a js module defines an array (I hope initialization is constant, the current component references to change the data in the module, switch components, data has been not the initial state, because there is no overloading, the page and I use the factory function to solve the problem

Replay:

index.html

 <script type="module">
     import { imgs } from './module1.js'
     console.log(imgs);
  </script>
Copy the code

module1.js

export const imgs = [
    {
      img: 'xxx'.isShow: false}, {img: 'xxx'.isShow: false}, {img: 'xxx'.isShow: false,},]Copy the code

If you make changes to the IMgs in HTML, it will affect Module1, so to avoid the use of json.parse (json.stringify (xx)), I used the factory function to return index.html in output

  <script type="module">
     import { imgs } from './module1.js'
     console.log(imgs());

  </script>
Copy the code

module1.js


  export const imgs = () = > {
    return[{img: 'xxx'.isShow: false}, {img: 'xxx'.isShow: false}, {img: 'xxx'.isShow: false,]}}Copy the code

6.2 Verify that import is loaded at compile time and CommonJS modules are loaded at run time(I was passed in the interview.)

From the above 2 we already know that there is a compile phase before the code is executed, that is, we need to parse our code into a computer aware language, AMD is run time load, engine processing import statement is compile time, how to verify,ES6 solved what problem

Here we test the ES6 module

module1.js

export default function test() {
  console.log('Welcome to pay attention to my public number -[program yuan love to talk]- small orange');
}

Copy the code

module2.js

export default function test() {
  console.log('Welcome to my public number -[program yuan love to talk]- big orange');
}
Copy the code

index.html

 <script type="module">
    const box = 1
    if(box === 1) {
      import test from './module1.js'
    } else {
      import test from './module2.js'
    }
    test()
  </script>
Copy the code

Error: Compile import because they use expressions, variables, and if structures. In static analysis (compilation), none of these syntaxes yields a value. This proves that the engine handles import statements at compile time

See CommonJS

node2.js

export default function test() {
  console.log('Welcome to my public number -[program yuan love to talk]- big orange');
}
Copy the code

node1.js

export default function test() {
  console.log('Welcome to pay attention to my public number -[program yuan love to talk]- small orange');
}
Copy the code

node0.js

const box = 1
let res
if (box === 1) {
  res = require('./node1.js');
} else {
  res = require('./node2.js');
}
console.log(res());
Copy the code

Proof that CommonJS is loaded at runtime

6.3 Why is import brought to the front

Because import statements are not parsed or executed at compile time when the engine processes them, it makes no sense for import statements to be placed in an if block of code, thus generating syntactic errors rather than run-time errors. That is, import and export commands can only be at the top level of a module, not in a code block (for example, in an if code block, or in a function).

If there are a lot of colorful shuttlecocks in a bucket, you must first pick out the blue one, and then consider whether to put the blue one on top when you put the shuttlecocks, which will improve the efficiency

6.4 Engine ProcessingimportWhat are the benefits of statements at compile time?

ES6 module dependencies are deterministic, independent of runtime state, and can be reliably statically analyzed, which is the basis for tree-shaking.

The so-called static analysis is not the execution of the code, from the literal analysis of the code, before ES6 modular, for example, we can dynamically require a module, only after the execution of the reference module, this can not be optimized through static analysis.

This is an important consideration in the design of ES6 Modules, and it’s why we don’t use CommonJS directly, it’s what makes tree-shaking possible, This is why rollup and Webpack 2 both use ES6 Module syntax to tree-shaking.

From the example of CommonJS in 6.1, we can see that I can decide what modules to introduce at runtime, so dynamic introduction cannot be used for static analysis, while module in ES6 is determined at compilation stage and can be analyzed.

7. Reference and citation

Require.js learn about ES6 javScript that you do not know, if there is any omission, I can fill in

8. Personal public account

Recently started to write a public, mainly some of the things happened in the front-end industry, and warm little story in life, working people is not easy, need to feel life, to love life, welcome to my attention, cage cage cage cage cage cage cage cage pay attention to me, thank you thank you! Is the public number is not to want you to pay attention to my nuggets account, you want to go!