webpack之commonjs模塊化原理

分析webpack是如何實現模塊化的webpack

準備

首先建立一個簡單入口模塊index.js和一個依賴模塊bar.jsweb

//index.js
'use strict';
var bar = require('./bar');

function foo() {
    bar();
}
複製代碼
//bar.js
'use strict';
exports.bar = function(){
    return 'bar';
}
複製代碼

webpack的配置以下segmentfault

var path = require("path");
module.exports = {
    mode: 'development',
    entry: path.join(__dirname, './src/index.js'),
    output: {
        path: path.join(__dirname, 'dist'),
        filename: 'index.js?v=[hash:8]',
    },
};
複製代碼

打包後獲得的代碼緩存

(function(modules) { // webpackBootstrap
/******/ 	// The module cache
/******/ 	var installedModules = {};
/******/
/******/ 	// The require function
/******/ 	function __webpack_require__(moduleId) {
/******/
/******/ 		// Check if module is in cache
/******/ 		if(installedModules[moduleId]) {
/******/ 			return installedModules[moduleId].exports;
/******/ 		}
/******/ 		// Create a new module (and put it into the cache)
/******/ 		var module = installedModules[moduleId] = {
/******/ 			i: moduleId,
/******/ 			l: false,
/******/ 			exports: {}
/******/ 		};
/******/
/******/ 		// Execute the module function
/******/ 		modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ 		// Flag the module as loaded
/******/ 		module.l = true;
/******/
/******/ 		// Return the exports of the module
/******/ 		return module.exports;
/******/ 	}
/******/
/******/
/******/ 	// expose the modules object (__webpack_modules__)
/******/ 	__webpack_require__.m = modules;
/******/
/******/ 	// expose the module cache
/******/ 	__webpack_require__.c = installedModules;
/******/
/******/ 	// define getter function for harmony exports
/******/ 	__webpack_require__.d = function(exports, name, getter) {
/******/ 		if(!__webpack_require__.o(exports, name)) {
/******/ 			Object.defineProperty(exports, name, { enumerable: true, get: getter });
/******/ 		}
/******/ 	};
/******/
/******/ 	// define __esModule on exports
/******/ 	__webpack_require__.r = function(exports) {
/******/ 		if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
/******/ 			Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
/******/ 		}
/******/ 		Object.defineProperty(exports, '__esModule', { value: true });
/******/ 	};
/******/
/******/ 	// create a fake namespace object
/******/ 	// mode & 1: value is a module id, require it
/******/ 	// mode & 2: merge all properties of value into the ns
/******/ 	// mode & 4: return value when already ns object
/******/ 	// mode & 8|1: behave like require
/******/ 	__webpack_require__.t = function(value, mode) {
/******/ 		if(mode & 1) value = __webpack_require__(value);
/******/ 		if(mode & 8) return value;
/******/ 		if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
/******/ 		var ns = Object.create(null);
/******/ 		__webpack_require__.r(ns);
/******/ 		Object.defineProperty(ns, 'default', { enumerable: true, value: value });
/******/ 		if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
/******/ 		return ns;
/******/ 	};
/******/
/******/ 	// getDefaultExport function for compatibility with non-harmony modules
/******/ 	__webpack_require__.n = function(module) {
/******/ 		var getter = module && module.__esModule ?
/******/ 			function getDefault() { return module['default']; } :
/******/ 			function getModuleExports() { return module; };
/******/ 		__webpack_require__.d(getter, 'a', getter);
/******/ 		return getter;
/******/ 	};
/******/
/******/ 	// Object.prototype.hasOwnProperty.call
/******/ 	__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
/******/
/******/ 	// __webpack_public_path__
/******/ 	__webpack_require__.p = "";
/******/
/******/
/******/ 	// Load entry module and return exports
/******/ 	return __webpack_require__(__webpack_require__.s = "./src/index.js");
})({

    "./src/bar.js":
    (function(module, exports, __webpack_require__) {
        "use strict";
        eval("\nexports.bar = function(){\n return 'bar';\n}\n\n//# sourceURL=webpack:///./src/bar.js?");
    }),
    "./src/index.js":
    (function(module, exports, __webpack_require__) {
        "use strict";
        eval("\nvar bar = __webpack_require__(/*! ./bar */ \"./src/bar.js\");\n\nfunction foo() {\n bar();\n}\n\n//# sourceURL=webpack:///./src/index.js?");
    })
});
複製代碼

分析

簡化打包後的文件bash

(function(modules) { 
    //模塊代碼
})({
    "./src/bar.js":
    (function(module, exports, __webpack_require__) {
        //bar.js代碼
    }),
    "./src/index.js":
    (function(module, exports, __webpack_require__) {
        //index.js代碼
    })
});
複製代碼
function(modules){
    //一、定義模塊緩存
    var installedModules = {};
    //二、webpack實現的require
    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;
        // 七、返回module.exports
        return module.exports;
    }
    __webpack_require__.m = modules;// 暴露模塊對象(__webpack_modules__)
    __webpack_require__.c = installedModules;//暴露模塊緩存
    __webpack_require__.d = function(exports, name, getter) {// 爲標準exports定義getter方法
        if(!__webpack_require__.o(exports, name)) {
            Object.defineProperty(exports, name, { enumerable: true, get: getter });
        }
    };
    __webpack_require__.r = function(exports) {// 在exports中定義__esModule
        if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
            Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
        }
        Object.defineProperty(exports, '__esModule', { value: true });
    };
    __webpack_require__.t = function(value, mode) {// 創造一個虛擬的命名空間
        if(mode & 1) value = __webpack_require__(value);
        if(mode & 8) return value;
        if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
        var ns = Object.create(null);
        __webpack_require__.r(ns);
        Object.defineProperty(ns, 'default', { enumerable: true, value: value });
        if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
            return ns;
    };
    __webpack_require__.n = function(module) {//getDefaultExport函數,用於與非標準模塊兼容
        var getter = module && module.__esModule ?
        function getDefault() { return module['default']; } :
        function getModuleExports() { return module; };
        __webpack_require__.d(getter, 'a', getter);
        return getter;
    };
    __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
    __webpack_require__.p = "";//公共路徑
    
    return __webpack_require__(__webpack_require__.s = "./src/index.js"); //加載入口模塊,並返回exports
}
複製代碼
  • 一、模塊對象做爲參數傳入IIFE函數
  • 二、IIFE首先定義了installedModules ,這個變量被用來緩存已加載的模塊
  • 三、定義了__webpack_require__ 這個函數,函數參數爲模塊的id。這個函數用來實現模塊的require
  • 四、若是沒有緩存,也就是第一次加載,則首先初始化模塊,並將模塊進行緩存
  • 五、而後調用模塊函數,也就是前面webpack對咱們的模塊的包裝函數,將module、module.exports和__webpack_require__做爲參數傳入。注意這裏作了一個動態綁定,將模塊函數的調用對象綁定爲module.exports,這是爲了保證在模塊中的this指向當前模塊
  • 六、調用完成後,模塊標記爲已加載
  • 七、返回模塊exports的內容

總結

實現exports和require,而後自動加載入口模塊,控制緩存模塊模塊化

參考文檔

webpack模塊化原理-commonjs函數

相關文章
相關標籤/搜索