博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
webpack 最简打包结果分析
阅读量:6048 次
发布时间:2019-06-20

本文共 5660 字,大约阅读时间需要 18 分钟。

原文链接:

现在的 webpack 不再是入门噩梦,过去 webpack 最让人心塞的莫过于配置文件,而 webpack4 诞生随之而来的是无配置 webpack。

使用 webpack4,至少只需要安装 webpack 和 webpack cli。所以大家完全可以自己打一个最简单的包,还能修改插件对比前后的区别。

npm i webpack webpack-cli -D 安装后,因为 webpack4 会默认 src 为入口目录,所以先新建 src/index.js

// src/index.jsimport { sth } from './shouldImport'import other from './shouldImport'let test = 'this is a variable'export default {  a: test + ',' + sth,  other,}

为了更了解 webpack  导入机制所以再新建 src/shouldImport.js

// src/shouldImport.jsexport let sth = 'something you need'export default {  others: '',}

然后运行 node_modules/.bin/webpack --mode development 即可在 dist/main.js 看到打包后的文件。

但是默认设置中模块文件会被 eval 包裹导致不便查看,所以需要再在设置做一点修改,把 devtool 属性改为 'source-map'

// 在根目录新建 webpack.config.js 文件module.exports = mode => {  if (mode === 'production') {    return {}  }  return {    devtool: 'source-map',  }}

然后再打包应该就能看到类似一下的文件结构,开发环境下打包得到的文件自带注释,理解起来不难:

;(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    // 执行模块函数(有点不懂为什么 this 要传入 module.exports)    modules[moduleId].call(      module.exports, // this      module, // 模块对象本身      module.exports, // 模块对象的 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  // Object.prototype.hasOwnProperty.call  __webpack_require__.o = function(object, property) {    return Object.prototype.hasOwnProperty.call(object, property)  }  // define getter function for harmony exports  // 为 ES6 export 定义 getter 函数  __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  // 于 export 定义 __esModule  __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  // value 是模块 id,必要  // mode & 2: merge all properties of value into the ns  // 合并 value 所有属性到 ns  // mode & 4: return value when already ns object  // ns 已经是对象时返回 value  // mode & 8|1: behave like require  // 表现如 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  // 用于兼容非 ES6 模块的 getDefaultExport 函数  __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  }  // __webpack_public_path__  __webpack_require__.p = ''  // Load entry module and return exports  // 加载入口模块并返回 export  return __webpack_require__((__webpack_require__.s = './src/index.js'))})({  './src/index.js':    /*! exports provided: default */    function(module, __webpack_exports__, __webpack_require__) {      'use strict'      __webpack_require__.r(__webpack_exports__) // 于 export 定义 __esModule      /* harmony import */      var _shouldImport__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(        './src/shouldImport.js'      )      let test = 'this is a variable'      /* harmony default export */      __webpack_exports__['default'] = {        a: test + ',' + _shouldImport__WEBPACK_IMPORTED_MODULE_0__['sth'],        other: _shouldImport__WEBPACK_IMPORTED_MODULE_0__['default'],      }    },  './src/shouldImport.js':    /*! exports provided: sth, default */    function(module, __webpack_exports__, __webpack_require__) {      'use strict'      __webpack_require__.r(__webpack_exports__)      /* harmony export (binding) */      __webpack_require__.d(__webpack_exports__, 'sth', function() {        return sth      })      let sth = 'something you need'      __webpack_exports__['default'] = {        others: '',      }    },})

源文件中的所有 importexport 都会转换为对应的辅助函数。

  • import 对应 __webpack_require__
  • export 对应 __webpack_exports__['default'] 直接赋值和 __webpack_require__.d

整理一下整个流程:

  1. 定义 __webpack_require__ 及其辅助函数
  2. 使用 __webpack_require__ 引入入口模块
  3. __webpack_require__ 函数载入模块,将模块放到模块缓存
  4. 调用模块

    1. 同样使用 __webpack_require__ 读取依赖(回到第 3 步)
    2. 运行模块内部功能
    3. 使用 __webpack_exports__['default'] 直接赋值和 __webpack_require__.d 输出
  5. 运行结束

转载地址:http://frxex.baihongyu.com/

你可能感兴趣的文章
Java基础String处理
查看>>
基于腾讯Angel的LDA*入选VLDB,超越微软LightLDA
查看>>
Java8特性——Lambda表达式
查看>>
mysql QPS TPS
查看>>
linux下apache+SVN搭建完美版
查看>>
阿里云新装数据库无法连接
查看>>
用java实现验证码
查看>>
导数中的最小化日志记录:背景和理论
查看>>
Git教程【译】(三)
查看>>
Jenkins入门总结
查看>>
并发,std::atomic
查看>>
jsp与php、asp的区别
查看>>
nali命令--输出IP地址显示地理信息
查看>>
RabbitMQ erlang "work queues"
查看>>
Global IP in Softlayer
查看>>
Myeclipse中搭建Android开发环境
查看>>
Windows Phone 8 锁屏背景与通知
查看>>
小蚂蚁学习数据结构(17)——树、二叉树性质、储存方式
查看>>
高性能WEB开发之JS、CSS的合并压缩
查看>>
raid卡与hba卡的区别
查看>>