美文网首页前端开发那些事儿
研究了一下 Webpack 打包原理,顺手挣了个 AirPods

研究了一下 Webpack 打包原理,顺手挣了个 AirPods

作者: 1024译站 | 来源:发表于2021-03-01 10:00 被阅读0次

    这些年,Webpack 基本成了前端项目打包构建的标配。关于它的原理和用法的文章在网上汗牛充栋,大家或多或少都看过一些。我也一样,大概了解过它的构建过程以及常用 loader 和 plugin 的配置、性能优化方法等等,仅限于“面试够用”的程度。在实际工作中,往往是配置好后就放一边了,没有遇到问题是不会再碰它的。

    我一直有个习惯(或者叫毛病),就是不太愿意花时间去研究暂时用不上的技术。我称其为“屠龙之技”:学会了屠龙的技术,可是找不到龙啊。这样的技术没有实际应用来强化,过不了多久就会荒废的。也因为这个,之前面试吃过很多亏,毕竟由于平台所限,工作中根本接触不到某些方面的技术。不过话又说回来,为了面试也要去学,硬着头皮的那种。

    扯远了,说回正题。前不久,网上有个哥们通过我的一篇博客找到我,让我帮他解决一个问题。这篇博客是关于如何在现有 Vue.js 项目里快速实现多语言切换的。他的项目也遇到同样的问题,但是他不懂代码,想付费求助。


    image.png

    按照我的方法,应该能很快完成需求。我大概估算了下工作量,报了个价。但是后面了解到的情况让我大跌眼镜:他的项目是打包好的,没有源码!说原来的开发不在了,都联系不上,找不到源码。要在没有源码的已有项目上加功能,写代码这么多年,还是第一次碰到。

    我那篇文章的方案,是重写 Vue.prototype.__patch__ 方法,拦截 DOM 渲染过程,将翻译后的文本替换上去。面对一坨可读性极差的压缩代码,还怎么写下去?当时他还没付款,我本打算放弃了。直到晚上睡觉前,这个问题一直盘旋在脑海里,挥之不去。难道我的方案有这么大的局限性?很不服气啊!

    没想到第二天,突然开窍了。这个问题的核心,不就是从压缩代码里找到 Vue 的引用吗?剩下的逻辑,都可以通过注入自己的 JS 代码来完成。

    明确了这个思路,就开始了压缩代码挖掘之旅。我们都知道,Vue 项目在打包构建后,会在 HTML 文件里注入几个 JS 文件,大概像这样:


    image.png

    其中的 vendor.xxx.js 就包含了 Vue.js 框架代码。但我们知道,这样构建出来的代码肯定是用了闭包,各个模块都被作用域屏蔽了,window下是访问不到这些模块的。可以试试在控制台输入 Vue ,会提示Uncaught ReferenceError: Vue is not defined

    这个时候就需要研究 Webpack 是怎么打包的了。这里的关键在 manifest.js 文件,它是 Webpack 的运行时代码,定义了一个webpackJsonp函数,代码简化后是这样的:

    (function(modules) {
      window["webpackJsonp"] = function webpackJsonpCallback(chunkIds, moreModules, executeModules) {
        var moduleId, result;
        for (moduleId in moreModules) {
          if (Object.prototype.hasOwnProperty.call(moreModules, moduleId)) {
            modules[moduleId] = moreModules[moduleId];
          }
        }
        if (executeModules) {
          for (i = 0; i < executeModules.length; i++) {
            result = __webpack_require__(executeModules[i]);
          }
        }
        return result;
      };
      var installedModules = {};
    
      function __webpack_require__(moduleId) {
        if (installedModules[moduleId]) {
          return installedModules[moduleId].exports;
        }
        var module = installedModules[moduleId] = {
          exports: {}
        };
        modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
        return module.exports;
      }
    })([]);
    

    打包后就是通过这个函数来加载各个模块的。因此,只要找到 Vue 这个模块被打包后的 ID,就能通过它来获取。再看看vendor.xxx.js这个文件内容:

    webpackJsonp([38], {
        "+abY": function(t, e, n) {
            "use strict";
            n("DmDj")("sup", function(t) {
                return function() {
                    return t(this, "sup", "", "")
                }
            })
        },
        "+fX/": function(t, e, n) {
            var r = n("awYD")
              , i = n("JE6n")
              , o = n("0U5H")("match");
            t.exports = function(t) {
                var e;
                return r(t) && (void 0 !== (e = t[o]) ? !!e : "RegExp" == i(t))
            }
        },
      "IvJb": function(t, e, n) {
          // 这就是 Vue 框架代码
      }
    )
    

    可以看到各个模块就是一个个的function。通过 Vue 框架里的一些关键字搜索,找到了 Vue 打包后的 ID 是IvJb。因此只要调用webpackJsonp函数就能获取 Vue变量:

    var vue = webpackJsonp([], {}, ['IvJb']);
    var __patch__ = vue.default.prototype.__patch__;
    vue.default.prototype.__patch__ = function () {
    var elm = __patch__.apply(this, arguments);
      var lang = getUrlParam('lang')
      if (lang) {
        //翻译DOM里的文本
        translate(elm, lang);
      }
      return elm;
    };
    

    关键问题解决了!通过同样的办法,还可以获取 axios ,把 axiosbaseUrl 改成了完整路径方便本地调试。剩下的工作就简单了,一是多语言文件文字翻译,那都是体力活,就交给那哥们自己干了。二是加一个语言切换菜单,这个也不难,原生 DOM 操作而已,再稍微调下样式就搞定了。

    前前后后花了不到一天时间,完成了这个看似不可能的任务。由此可见,了解工具和框架的底层原理,对于解决特定问题有着决定性的作用。当然,Webpack 功能非常强大,底层逻辑比这里说的复杂多了,我也没有继续深入研究。或许下次碰到问题时又是一次契机。

    关于多语言切换的方案,参考我之前写的博客:现有 Vue.js 项目快速实现多语言切换的一种思路

    本文首发于公众号 1024译站,这里不让发二维码。简书的未来,肉眼可见。

    相关文章

      网友评论

        本文标题:研究了一下 Webpack 打包原理,顺手挣了个 AirPods

        本文链接:https://www.haomeiwen.com/subject/itdmfltx.html