preface

Speaking of lazy route loading, we quickly know how to implement it, but when asked about the principle of lazy route loading, afraid of some friends are at a loss. Let’s understand the principle of lazy route loading.

Route lazy loading can also be called routing component lazy loading, most commonly implemented with import().

function load(component) {
    return () => import(`views/${component}`)
}
Copy the code

After compiling and packaging through Webpack, the code of each routing component will be divided into a JS file. These JS files will not be loaded during initialization, and the corresponding JS file will be loaded only when the routing component is activated.

After Webpack is compiled, how to load the corresponding js file of the routing component on demand is implemented.

First, preparation

1. Build the project

To understand the principle of lazy routing loading, it is recommended to start with the simplest project and build a project using Vue Cli3 that contains only one routing component. Introduce only vue-router in main.js and nothing else.

main.js

import Vue from 'vue'; import App from './App.vue'; import Router from 'vue-router'; Vue.use(Router); Function load(component) {return () => import(' views/${component} ')} // Const router = new router ({mode:  'history', base: process.env.BASE_URL, routes: [ { path: '/', name: 'home', component: load('Home'), meta: { title: }},]}); new Vue({ router, render: h => h(App) }).$mount('#app')Copy the code

views/Home.vue

<template> <div> {{tip}} <div> </template> <script> export default {data(){return {tip :' welcome to Vue project '}}} </script>Copy the code

2, webpackChunkName

Use webpackChunkName to make the compiled js file name and routing component one to one, modify the load function.

function load(component) {
    return () => import(/* webpackChunkName: "[request]" */ `views/${component}`)
}
Copy the code

3, remove code compression obfuscation

Remove code compression obfuscations and make it easier for us to read compiled and packaged code. In vue.config.js

module.exports={ chainWebpack:config => { config.optimization.minimize(false); }},Copy the code

4, NPM run build

Run the NPM run build command to compile the packaged dist file structure as shown below

Home.67f3cd34.js is the js file compiled and packaged by the routing component home.vue.

2. Analyze index.html

As you can see above, the link is used to define the relationship between home.js, app.js, and chunk-vision.js resources and the Web client.

  • ref=preload: Tell the browser to load this resource for me in advance.
  • rel=prefetch: Tell the browser to load this resource when it is free.
  • as=script: tells the browser that the resource is a script, raising the load priority.

Then I load the chunk-filents.js and app.js resources into the body. You can see that these two JS resources are loaded when the Web client is initialized.

Analyze chunk-pds.js

Chunk-pdf.js can be called the project common module collection, and the code is simplified as follows,

(window["webpackJsonp"] = window["webpackJsonp"] || []).push([["chunk-vendors"],{ "01 f9" (function (module, exports, __webpack_require__) / / omit} {...... / / omit}])Copy the code

As you can see from the code, executing chunk-vision.js simply pushes the following array into the window[“webpackJsonp”]. The second item in the array is an object, and each value of the object is a function expression that is not executed. That’s the end of it, of course not, we took the window[“webpackJsonp”] and went to app.js.

Analyze app.js

App.js can be called the project’s entry file.

App.js is a self-executing function. Search window[“webpackJsonp”] to find the following code.

(function(modules){// omit... var jsonpArray = window["webpackJsonp"] = window["webpackJsonp"] || []; var oldJsonpFunction = jsonpArray.push.bind(jsonpArray); jsonpArray.push = webpackJsonpCallback; jsonpArray = jsonpArray.slice(); for (var i = 0; i < jsonpArray.length; i++) webpackJsonpCallback(jsonpArray[i]); var parentJsonpFunction = oldJsonpFunction; / / to omit... }({ 0:(function(module, exports, __webpack_require__) { module.exports = __webpack_require__("56d7"); }) // Omit... }))Copy the code
  • The firstwindow["webpackJsonp"]Assigned tojsonpArray.
  • thejsonpArraythepushMethod tooldJsonpFunction.
  • withwebpackJsonpCallbackFunction to interceptjsopArraythepushMethod, that is, callwindow["webpackJsonp"]thepushMethods are executedwebpackJsonpCallbackFunction.
  • willjsonpArrayShallow copy and assign tojsonpArray.
  • Because of the chunk-pds.js implementationwindow["webpackJsonp"].pushwhenpushThe method has not beenwebpackJsonpCallbackFunction intercept, so loopjsonpArray, passing each item as a parameterwebpackJsonpCallbackFunction and call.
  • willjsonpArraythepushMethod is assigned toparentJsonpFunction.

1. WebpackJsonpCallback

Next, let’s look at the webpackJsonpCallback function.

(function(modules){ function webpackJsonpCallback(data) { var chunkIds = data[0]; var moreModules = data[1]; var executeModules = data[2]; var moduleId, chunkId, i = 0, resolves = []; for (; i < chunkIds.length; i++) { chunkId = chunkIds[i]; if (Object.prototype.hasOwnProperty.call(installedChunks, chunkId) && installedChunks[chunkId]) { resolves.push(installedChunks[chunkId][0]); } installedChunks[chunkId] = 0; } for (moduleId in moreModules) { if (Object.prototype.hasOwnProperty.call(moreModules, moduleId)) { modules[moduleId] = moreModules[moduleId]; } } if (parentJsonpFunction) parentJsonpFunction(data); while (resolves.length) { resolves.shift()(); } deferredModules.push.apply(deferredModules, executeModules || []); return checkDeferredModules(); }; var installedChunks = { "app": 0 }; / / to omit... }({ 0:(function(module, exports, __webpack_require__) { module.exports = __webpack_require__("56d7"); }) // Omit... }))Copy the code

To understand what the webpackJsonpCallback function does, you need to understand the functions of modules, installedChunks, and deferredModules.

  • A module is an arbitrary block of code, and a chunk is a collection of Modules grouped during WebPack processing.
  • modulesCache all module (code block) callsmodulesTo execute the code in the module.
  • installedChunksCache the loading status of all chunksinstalledChunks[chunk]0 indicates that the chunk has been loaded.
  • deferredModulesEach item is also an array, for example[module,chunk1,chunk2,chunk3]Chunk1, chunk2, and chunk3 must be loaded before a module is executed.

If (parentJsonpFunction) parentJsonpFunction(data) = parentJsonpFunction The parentJsonpFunction call actually pushes the chunk push parameters into the window[“webpackJsonp”] array.

For example, now the project has two entries, app.js and app1.js. App.js cache some modules, and app1.js can call these modules using window[“webpackJsonp”].

for (var i = 0; i < jsonpArray.length; i++) webpackJsonpCallback(jsonpArray[i]);
Copy the code

To see if the webpackJsonpCallback function is a lot clearer, take a look at checkDeferredModules.

2, checkDeferredModules

var deferredModules = []; var installedChunks = { "app": 0 } function checkDeferredModules() { var result; for (var i = 0; i < deferredModules.length; i++) { var deferredModule = deferredModules[i]; var fulfilled = true; for (var j = 1; j < deferredModule.length; j++) { var depId = deferredModule[j]; if (installedChunks[depId] ! == 0) fulfilled = false; } if (fulfilled) { deferredModules.splice(i--, 1); result = __webpack_require__(__webpack_require__.s = deferredModule[0]); } } return result; }Copy the code
  • cycledeferredModules, create a variablefulfilledsaiddeferredModuleThe chunk loading condition in,trueIndicates that all loads are complete,falseIndicates that the load is not complete.
  • fromj=1Start cycledeferredModuleChunk, becausedeferredModule[0]Yes module, ifinstalledChunks[chunk]! = = 0“, then the chunk has not been loadedfulfilledSet tofalse. Return result at the end of the loop.
  • By the cycledeferredModuleAnd determine the loading state of the chunk,fulfilledIf true is still true__webpack_require__Function,deferredModule[0](Module) is passed in as a parameter.
  • deferredModules.splice(i--, 1), remove any conforming deferredModule and subtract one from I, wherei--I use I first, and then I subtract one.

Because deferredModules are [] in the webpackJsonpCallback function, go back to the body function and continue.

deferredModules.push([0, "chunk-vendors"]);
return checkDeferredModules();
Copy the code

Following the above logic, __webpack_require__(0) is executed, so take a look at __webpack_require__.

3. __webpack_require__ function

var installedModules = {};
function __webpack_require__(moduleId) {
    if (installedModules[moduleId]) {
        return installedModules[moduleId].exports;
    }
    var module = installedModules[moduleId] = {
        i: moduleId,
        l: false,
        exports: {}
    };
    modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
    module.l = true;
    return module.exports;
}
Copy the code

From the code, __webpack_require__ is a method to execute the module.

  • installedModulesUse to cache module execution status.
  • Modules via moduleId (inwebpackJsonpCallbackThe call method is used to retrieve the corresponding module.
  • Assign the execution result to module.exports and return.

So __webpack_require__(0) executes the following code.

(function (module, exports, __webpack_require__) {
    module.exports = __webpack_require__("56d7");
}),
Copy the code

Using __webpack_require__ to execute module 56d7, we find the corresponding module and continue to look at the key code snippet.

function load(component) { return function () { return __webpack_require__("9dac")("./".concat(component)); }; } routes = [{path: '/', name: 'home', component: load(' home'), meta: {title: 'home'}}, {path: '*', redirect: { path: '/' } }];Copy the code

If you’re familiar with this, this is where the routing is configured. Load is also used as a function to load the routing component, which is performed using the method __webpack_require__(“9dac”) returned. Let’s take a look at __webpack_require__(“9dac”).

(function (module, exports, __webpack_require__) { var map = { "./Home": [ "bb51", "Home" ], "./Home.vue": [ "bb51", "Home" ] }; function webpackAsyncContext(req) { if (! __webpack_require__.o(map, req)) { return Promise.resolve().then(function () { var e = new Error("Cannot find module '" + req + "'"); e.code = 'MODULE_NOT_FOUND'; throw e; }); } var ids = map[req], id = ids[0]; return __webpack_require__.e(ids[1]).then(function () { return __webpack_require__(id); }); } webpackAsyncContext.keys = function webpackAsyncContextKeys() { return Object.keys(map); }; webpackAsyncContext.id = "9dac"; module.exports = webpackAsyncContext; })Copy the code

WebpackAsyncContext function

The key function is webpackAsyncContext. When calling load(‘Home’), req is ‘./Home’ and __webpack_require__.o is

__webpack_require__.o = function (object, property) {
    return Object.prototype.hasOwnProperty.call(object, property);
};
Copy the code

/Home: Cannot find module ‘./Home’ : Cannot find module ‘./Home’ : Cannot find module ‘. The __webpack_require__.e method is executed with the argument Home.

5,webpack_requireE. method

var installedChunks = { "app": 0 } __webpack_require__.p = "/"; function jsonpScriptSrc(chunkId) { return __webpack_require__.p + "js/" + ({ "Home": "Home" }[chunkId] || chunkId) + "." + { "Home": "37ee624e" }[chunkId] + ".js" } __webpack_require__.e = function requireEnsure(chunkId) { var promises = []; var installedChunkData = installedChunks[chunkId]; if (installedChunkData ! == 0) { if (installedChunkData) { promises.push(installedChunkData[2]); } else { var promise = new Promise(function (resolve, reject) { installedChunkData = installedChunks[chunkId] = [resolve, reject]; }); promises.push(installedChunkData[2] = promise); var script = document.createElement('script'); var onScriptComplete; script.charset = 'utf-8'; script.timeout = 120; if (__webpack_require__.nc) { script.setAttribute("nonce", __webpack_require__.nc); } script.src = jsonpScriptSrc(chunkId); var error = new Error(); OnScriptComplete = function (event) {// Avoid IE memory leak. script.onerror = script.onload = null; clearTimeout(timeout); var chunk = installedChunks[chunkId]; if (chunk ! == 0) { if (chunk) { var errorType = event && (event.type === 'load' ? 'missing' : event.type); var realSrc = event && event.target && event.target.src; error.message = 'Loading chunk ' + chunkId + ' failed.\n(' + errorType + ': ' + realSrc + ')'; error.name = 'ChunkLoadError'; error.type = errorType; error.request = realSrc; chunk[1](error); } installedChunks[chunkId] = undefined; }}; var timeout = setTimeout(function () { onScriptComplete({ type: 'timeout', target: script }); }, 120000); script.onerror = script.onload = onScriptComplete; document.head.appendChild(script); } } return Promise.all(promises); };Copy the code

The __webpack_require__.e method is at the heart of lazy loading, and it handles three things.

  • Use JSONP mode to load the JS file corresponding to the route, also called chunk.
  • Set the three states of the chunk loading and cache theminstalledChunksTo prevent repeated loading of the chunk.
  • Handle the chunk load timeout and load error scenarios.

Three states of the chunk loading

  • installedChunks[chunkId]for0“Indicates that the chunk has been loaded.
  • installedChunks[chunkId]forundefined“, indicating that the chunk fails to be loaded, the loading times out, or the chunk is never loaded.
  • installedChunks[chunkId]forPromiseObject that indicates that the chunk is being loaded.

Chunk load timeout processing

script.timeout = 120;
var timeout = setTimeout(function () {
    onScriptComplete({ type: 'timeout', target: script });
}, 120000);
Copy the code

Script. timeout = 120 indicates that the chunk has timed out if the chunk is not loaded after 120 seconds. After 120 seconds, execute onScriptComplete({type: ‘timeout’, target: script}).

Take a look at the onScriptComplete function

Var onScriptComplete = function (event) { script.onerror = script.onload = null; clearTimeout(timeout); var chunk = installedChunks[chunkId]; if (chunk ! == 0) { if (chunk) { var errorType = event && (event.type === 'load' ? 'missing' : event.type); var realSrc = event && event.target && event.target.src; error.message = 'Loading chunk ' + chunkId + ' failed.\n(' + errorType + ': ' + realSrc + ')'; error.name = 'ChunkLoadError'; error.type = errorType; error.request = realSrc; chunk[1](error); } installedChunks[chunkId] = undefined; }};Copy the code

At this point chunkId is Home, the load is home.js, and the code is

(window["webpackJsonp"] = window["webpackJsonp"] || []).push([["Home"],{ "bb51":(function(module, __webpack_exports__, __webpack_require__){// omit...})}])Copy the code

Windows [“webpackJsonp”] push is intercepted by webpackJsonpCallback. If home.js loads successfully, it will be automatically executed, and then webpackJsonpCallback will be executed. InstalledChunks [chunkId] = 0; Sets the installedChunks[‘Home’] value to 0.

In other words, if the home.js load times out, it cannot be executed, and the installedChunks[‘Home’] value cannot be set to 0, so the installedChunks[‘Home’] value is still a Promise object. This leads to the following code execution, and finally chunk[1](error) throws the error.

var chunk = installedChunks[chunkId];
if(chunk!==0){
    if(chunk){
        var errorType = event && (event.type === 'load' ? 'missing' : event.type);
        var realSrc = event && event.target && event.target.src;
        error.message = 'Loading chunk ' + chunkId 
        + ' failed.\n(' + errorType + ': ' + realSrc + ')';
        error.name = 'ChunkLoadError';
        error.type = errorType;
        error.request = realSrc;
        chunk[1](error);
    }
}
Copy the code

Chunk [1] is the reject function, which is assigned in the following code.

var promise = new Promise(function (resolve, reject) {
    installedChunkData = installedChunks[chunkId] = [resolve, reject];
});
Copy the code

Description Failed to load the chunk

There are two types of loading failures: one is that the home. js resource fails to be loaded; the other is that the home. js resource fails to be loaded successfully, but the home. js code fails due to an error

script.onerror = script.onload = onScriptComplete;
Copy the code

This is handled in the same way as the load timeout.

__webpack_require__.e finally returns a Promise object. So let’s go back to the webpackAsyncContext function

return __webpack_require__.e(ids[1]).then(function () {
    return __webpack_require__(id);
});
Copy the code

After __webpack_require__.e(ids[1]) succeeds, __webpack_require__(ID) is executed; , and the ID is BB51. So it’s back to the __webpack_require__ function. As mentioned earlier, the __webpack_require__ function is used to execute modules. The NOdule with ID BB51 is inside home.js, with the following code in the webpackJsonpCallback function

function webpackJsonpCallback(data) { var moreModules = data[1]; for (moduleId in moreModules) { if (Object.prototype.hasOwnProperty.call(moreModules, moduleId)) { modules[moduleId] = moreModules[moduleId]; }}}Copy the code

Analyze home.js

Home.js

(window["webpackJsonp"] = window["webpackJsonp"] || []).push([["Home"],{ "bb51":(function(module, __webpack_exports__, __webpack_require__){// omit...})}])Copy the code

{“bb51”:(function(module, __webpack_exports__, __webpack_require__){})},

Loop moreModules and cache modules from home.js into modules in app.js.

Look at this code in the __webpack_require__ function

modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
Copy the code

This executes the module in home.js, which has a series of methods to render the page, rendering the home.vue routing component page.

At this point the entire process of lazy loading of the routing component ends, and details how to load the chunk and execute the Module.