新聞中心
這些年,Webpack 基本成了前端項(xiàng)目打包構(gòu)建的標(biāo)配。關(guān)于它的原理和用法的文章在網(wǎng)上汗牛充棟,大家或多或少都看過(guò)一些。我也一樣,大概了解過(guò)它的構(gòu)建過(guò)程以及常用 loader 和 plugin 的配置、性能優(yōu)化方法等等,僅限于“面試夠用”的程度。在實(shí)際工作中,往往是配置好后就放一邊了,沒(méi)有遇到問(wèn)題是不會(huì)再碰它的。

我一直有個(gè)習(xí)慣(或者叫毛病),就是不太愿意花時(shí)間去研究暫時(shí)用不上的技術(shù)。我稱其為“屠龍之技”:學(xué)會(huì)了屠龍的技術(shù),可是找不到龍啊。這樣的技術(shù)沒(méi)有實(shí)際應(yīng)用來(lái)強(qiáng)化,過(guò)不了多久就會(huì)荒廢的。也因?yàn)檫@個(gè),之前面試吃過(guò)很多虧,畢竟由于平臺(tái)所限,工作中根本接觸不到某些方面的技術(shù)。不過(guò)話又說(shuō)回來(lái),為了面試也要去學(xué),硬著頭皮的那種。
扯遠(yuǎn)了,說(shuō)回正題。前不久,網(wǎng)上有個(gè)哥們通過(guò)我的一篇博客找到我,讓我?guī)退鉀Q一個(gè)問(wèn)題。這篇博客是關(guān)于如何在現(xiàn)有 Vue.js 項(xiàng)目里快速實(shí)現(xiàn)多語(yǔ)言切換的。他的項(xiàng)目也遇到同樣的問(wèn)題,但是他不懂代碼,想付費(fèi)求助。
按照我的方法,應(yīng)該能很快完成需求。我大概估算了下工作量,報(bào)了個(gè)價(jià)。但是后面了解到的情況讓我大跌眼鏡:他的項(xiàng)目是打包好的,沒(méi)有源碼!說(shuō)原來(lái)的開(kāi)發(fā)不在了,也聯(lián)系不上,找不到源碼。要在沒(méi)有源碼的已有項(xiàng)目上加功能,寫代碼這么多年,還是第一次碰到。
我那篇文章的方案,是重寫 Vue.prototype.__patch__ 方法,攔截 DOM 渲染過(guò)程,將翻譯后的文本替換上去。面對(duì)一坨可讀性極差的壓縮代碼,還怎么寫下去?當(dāng)時(shí)他還沒(méi)付款,我本打算放棄了。直到晚上睡覺(jué)前,這個(gè)問(wèn)題一直盤旋在腦海里,揮之不去。難道我的方案有這么大的局限性?很不服氣啊!
沒(méi)想到第二天,突然開(kāi)竅了。這個(gè)問(wèn)題的核心,不就是從壓縮代碼里找到 Vue 的引用嗎?剩下的邏輯,都可以通過(guò)注入自己的 JS 代碼來(lái)完成。
明確了這個(gè)思路,就開(kāi)始了壓縮代碼挖掘之旅。我們都知道,Vue 項(xiàng)目在打包構(gòu)建后,會(huì)在 HTML 文件里注入幾個(gè) JS 文件,大概像這樣:
其中的 vendor.xxx.js 就包含了 Vue.js 框架代碼。但我們知道,這樣構(gòu)建出來(lái)的代碼肯定是用了閉包,各個(gè)模塊都被作用域屏蔽了,window下是訪問(wèn)不到這些模塊的??梢栽囋囋诳刂婆_(tái)輸入 Vue ,會(huì)提示Uncaught ReferenceError: Vue is not defined。
這個(gè)時(shí)候就需要研究 Webpack 是怎么打包的了。這里的關(guān)鍵在 manifest.js 文件,它是 Webpack 的運(yùn)行時(shí)代碼,定義了一個(gè)webpackJsonp函數(shù),代碼簡(jiǎn)化后是這樣的:
- (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;
- }
- })([]);
打包后就是通過(guò)這個(gè)函數(shù)來(lái)加載各個(gè)模塊的。因此,只要找到 Vue 這個(gè)模塊被打包后的 ID,就能通過(guò)它來(lái)獲取。再看看vendor.xxx.js這個(gè)文件內(nèi)容:
- 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 框架代碼
- }
- )
可以看到各個(gè)模塊就是一個(gè)個(gè)的function。通過(guò) Vue 框架里的一些關(guān)鍵字搜索,找到了 Vue 打包后的 ID 是IvJb。因此只要調(diào)用webpackJsonp函數(shù)就能獲取 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;
- };
關(guān)鍵問(wèn)題解決了!通過(guò)同樣的辦法,還可以獲取 axios ,把 axios 的baseUrl 改成了完整路徑方便本地調(diào)試。剩下的工作就簡(jiǎn)單了,一是多語(yǔ)言文件文字翻譯,那都是體力活,就交給那哥們自己干了。二是加一個(gè)語(yǔ)言切換菜單,這個(gè)也不難,原生 DOM 操作而已,再稍微調(diào)下樣式就搞定了。
前前后后花了不到一天時(shí)間,完成了這個(gè)看似不可能的任務(wù)。由此可見(jiàn),了解工具和框架的底層原理,對(duì)于解決特定問(wèn)題有著決定性的作用。
當(dāng)然,Webpack 功能非常強(qiáng)大,底層邏輯比這里說(shuō)的復(fù)雜多了,我也沒(méi)有繼續(xù)深入研究。或許下次碰到問(wèn)題時(shí)又是一次契機(jī)呢。
本文轉(zhuǎn)載自微信公眾號(hào)「1024譯站」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系1024譯站公眾號(hào)。
當(dāng)前標(biāo)題:研究了一下 Webpack 打包原理,順手掙了個(gè) AirPods Pro
標(biāo)題路徑:http://m.5511xx.com/article/dpojpcp.html


咨詢
建站咨詢
