• Webpack原理 如何打包,看懂这篇文章就够了,面试必备技能


    前言

    作为一个前端“攻城狮”,Webpack 再熟悉不过了,Webpack 能做的事太多了,可以将所有资源(包括 JS,TS,JSX,图像,字体和 CSS 等)打包后置于依赖关系中,使你可以按照需求引用依赖来使用资源。Webpack 很出色的完成了转译前端多种文件资源,分析复杂模块依赖的工作,并且我们还可以自定义 loader,自由的加载我们自己的资源,那 Webpack 是如何实现打包的呢?今天来我们一起来看下。

    想要知道 Webpack 打包原理的我们需要提前知道两个知识点

    1、什么是 require?

    说到 require 首先想到的可能就是 import,import 是 es6 的一个语法标准,

    ​ – require 是运行时调用,因此 require 理论上可以运用在代码的任何地方;

    ​ – import 是编译时调用,因此必须放在文件开头;

    在我们使用 Webpack 进行编译的时候会使用 babel 把 import 转译成 require,在 CommonJS 中,有一个全局性方法 require(),用于加载模块, AMD、CMD 也采用的 require 方式来引用。

    例如:

    -   var add = require('./a.js');
    -   add(1,2) 
    
    • 1
    • 2

    简单看来 require 其实就是一个函数,引用的 ./a.js 只是函数的一个参数。

    2、什么是 exports?

    在这里我们可以认为 exports 是一个对象,MDN export 可以看下具体用法。

    了解了require 和 exports,接下来我们就可以开始打包

    我们先看看下面我们打包后的代码结构,我们可以发现经过打包后会出现 require 和 exports。

    并不是所有的浏览器都能执行 require exports,必须自己去实现一下 require 和 exports 才能保证代码的正常运行。打包后的代码就是一个自执行函数,参数有依赖信息,以及文件的 code,执行的函数体通过 eval 执行 code。

    \

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-d3DKn3Yd-1656592213370)(https://p1-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/1f0f7509f8bf446692a47a0636c326e3~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-WsUUWpfA-1656592213371)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0aefd10939954739add6cca4cf1d36b4~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    第一步:编写我们的配置文件

    配置文件中配置了我们打包的入口 entry 以及打包后的出口 output 为后面的生成文件做好准备。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ft95D2d7-1656592213371)(https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/cfbef2262ffe4a9eaddb6a7f8e9f894e~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    第二步:模块分析

    整体思路:可以总结来说就是利用 fs 文件读取入口文件 通过 AST 获取到 import 依赖的文件的路径,如果依赖文件 依然有依赖一直递归下去直至依赖分析清楚,维护在一个 map 里面。

    细节拆解:有人会有疑惑为什么用 AST 因为 AST 天生有这个功能,它的 ImportDeclaration 能帮我们快速过滤出 import 语法,当然用正则匹配也是可以的,毕竟文件读取完就是一个字符串,通过编写牛逼的正则获取文件依赖路径,但是不够 elegant。

    step1:新建 index.js,a.js,b.js 依赖关系如下

    index.js文件

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-B2OWBFE0-1656592213372)(https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/950ce688ab7840e292a987b3fd1f199f~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    step2:编写 Webpack

    模块分析:利用 AST 的 @babel/parser 将文件读取的字符串转换成 AST 树,@babel/traverse 进行语法分析,利用 ImportDeclaration 过滤出 import 找出文件依赖。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-euiT80y3-1656592213372)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/56820e1f0920494685ced17af2a0f54d~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-rTHlSB5p-1656592213372)(https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/1f56ecff115f4040a2c3657258182eb7~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    利用递归或是循环逐个 import 文件进行依赖分析,这块注意,我们是使用 for 循环实现了分析所有依赖,之所以循环可以分析所有依赖,注意 modules 的长度是变化的,当有依赖的时候 .modules.push 新的依赖,modules.length 就会变化。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-cZna50a8-1656592213372)(https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/7c05e875f3594960b88f781d4fba9df5~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    第三步:编写 WebpackBootstrap 函数+生成输出文件

    编写 WebpackBootstrap 函数:这里我们需要做的首先是 WebpackBootstrap 函数,编译后我们源代码的 import 会被解析成 require 浏览器既然不认识 require ,那我们就先声明它,毕竟 require 就是一个方法,在编写函数的时候还需要注意的是作用域隔离,防止变量污染。我们代码中 exports 也需要我们声明一下,保证代码在执行的时候 exports 已经存在。

    生成输出文件:生成文件的地址我们在配置文件已经写好了,再用 fs.writeFileSync 写入到输出文件夹即可。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dCAN6yWJ-1656592213373)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/a7bb5c44ca1f4c00bff3c2262fbc0deb~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0JwAth6d-1656592213373)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/65133cf3a4bb4f019b874e6ce4748b00~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    第四步:分析执行顺序

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-OQMOwUrN-1656592213373)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/f961e988f4724f76a1833bec65230662~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    我们可以在浏览器的控制台运行一下打包后的结果,如果能正常应该会打印出 hello Webpack。

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bucV5D2C-1656592213373)(https://p9-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/96458c0840cb4a9dbd183168864e3859~tplv-k3u1fbpfcp-zoom-in-crop-mark:1956:0:0:0.image?)]

    总结

    通过以上的分析,我们应该对 Webpack 的大概流程有基本的了解,利用 AST 去解析代码只是本次演示的一种方式,不是 Webpack 的真实实现,Webpack 他自己有自己的 AST 解析方式,万变不离其宗都是拿到模块依赖,Webpack 生态是很完整,有兴趣的童鞋可以考虑以下三个问题:

    • 如果出现组件循环引用那又应该如何处理?
    • Webpack 是如何加载 loader 的?
    • 犹大大极力推荐的 vite 可以实现按需打包,大大降低开发时候打包速度,如果是 webapck 又是应该如何实现?

    \

    参考

    zoo.team/article/web…

  • 相关阅读:
    Mybatis 缓存原理
    NVIDIA 7th SkyHackathon(六)Tao 目标检测模型训练与评估
    开曼群岛:Web3企业的乐园
    【第一阶段:java基础】第6章:面向对象编程基础(P191-P227)
    Spring Boot 使用Knife4j
    Spring的事务控制-编程式事务控制相关对象
    头像Css
    【面试宝藏】Redis 常见面试题解析
    企业做软文推广的三大错误有哪些?媒介盒子为您解答
    理德外汇名人故事:全球第一理财师——苏茜·欧曼
  • 原文地址:https://blog.csdn.net/web2022050903/article/details/125547774