深析webpack的打包流程和原理

webpack 是如何实现打包的?下面本篇文章就来带大家深入了解一下webpack 打包原理,希望对大家有所帮助!

深析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)

登录后复制

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

2、什么是 exports?

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

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

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

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

1.png

总体设计图如下:

2.png

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

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

const path = require("path");module.exports = {  entry: "./src/index.js",  output: {    path: path.resolve(__dirname, "./dist"),//打包后输出的文件地址,需要绝对路径因此需要path    filename:"main.js"  },  mode:"development"

登录后复制

第二步:模块分析

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

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

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

index.js文件

import { str } from "./a.js";console.log(`${str} Webpack`)

登录后复制

a.js文件

import { b} from "./b.js"export const str = "hello"

登录后复制

b.js 文件

export const b="bbb"

登录后复制

step2:编写 Webpack

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

    const content = fs.readFileSync(entryFile, "utf-8");    const ast = parser.parse(content, { sourceType: "module" });      const dirname = path.dirname(entryFile);    const dependents = {};    traverse(ast, {      ImportDeclaration({ node }) {        // 过滤出import        const newPathName = "./" + path.join(dirname, node.source.value);        dependents[node.source.value] = newPathName;      }    })    const { code } = transformFromAst(ast, null, {      presets: ["@babel/preset-env"]    })    return {      entryFile,      dependents,      code    }

登录后复制

结果如下:

3.png

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

 for (let i = 0; i < this.modules.length; i++) {      const item = this.modules[i];      const { dependents } = item;      if (dependents) {        for (let j in dependents) {          this.modules.push(this.parse(dependents[j]));        }      }    }

登录后复制

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

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

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

  file(code) {    const filePath = path.join(this.output.path, this.output.filename)    const newCode = JSON.stringify(code);    // 生成bundle文件内容    const bundle = `(function(modules){      function require(module){        function pathRequire(relativePath){          return require(modules[module].dependents[relativePath])        }        const exports={};        (function(require,exports,code){          eval(code)        })(pathRequire,exports,modules[module].code);        return exports      }      require('${this.entry}')    })(${newCode})`;      //  WebpackBoostrap    // 生成文件。放入dist 目录    fs.writeFileSync(filePath,bundle,'utf-8')  }

登录后复制

4.png

第四步:分析执行顺序

5.png

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

6.png

总结

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

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

更多编程相关知识,请访问:编程视频!!

以上就是深析webpack的打包流程和原理的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2699017.html

(0)
上一篇 2025年3月7日 18:54:29
下一篇 2025年2月23日 14:53:29

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • 深入解析webpack的五个核心概念

    webpack 是一种前端资源构建工具,一个静态模块打包器(module bundler);webpack有5大核心概念(入口、输出、loader、插件、模式) ,下面本篇文章就来带大家深入了解一下,希望对大家有所帮助! webpack 五…

    2025年3月7日 编程技术
    200
  • webpack核心概念之输出(Output)

    将所有的资源(assets)归拢在一起后,还需要告诉 webpack 在哪里打包应用程序。webpack 的 output 属性描述了如何处理归拢在一起的代码(bundled code)。下面本篇文章就来带大家深入了解一下webpack核心…

    2025年3月7日 编程技术
    200
  • webpack核心概念之入口配置(entry)

    在一切流程的最开始,我们需要指定一个或多个入口(entry),也就是告诉webpack具体从源码目录下的哪个文件开始打包。下面本篇文章就来带大家深入了解一下webpack核心概念中的入口配置(entry),希望对大家有所帮助! 如果把工程中…

    2025年3月7日 编程技术
    200
  • 聊聊webpack中怎么压缩打包html资源

    webpack中怎么压缩打包html资源?下面本篇文章就来给大家简单介绍一下webpack压缩打包html资源的方法,希望对大家有所帮助! 为什么需要打包html资源 写代码时引入的是src下面的js文件,经过webpack打包后,形成了一…

    2025年3月7日
    200
  • 【整理分享】一些webpack面试题(附答案解析)

    谈谈你对Webpack的理解 1.Webpack是什么? webpack 是一个静态模块打包器,当 webpack 处理应用程序时,会递归构建一个依赖关系图,其中包含应用程序需要的每个模块,然后将这些模块打包成一个或多个 bundle。 w…

    2025年3月7日
    200
  • 如何利用React和Webpack实现前端代码的模块化打包

    如何利用React和Webpack实现前端代码的模块化打包 引言:在前端开发中,随着项目的复杂性增加,代码的管理和维护变得越来越困难,因此模块化打包成为了必不可少的工具。React作为一款流行的前端框架,通过组件化开发的思想,大大简化了复杂…

    2025年3月7日
    200
  • 使用vue2.x+webpack如何搭建前端项目

    本文给大家介绍了vue2.x、webpack、vuex、sass+axios、elementui等快速搭建前端项目框架的详细操作方法,需要的跟着学习下吧。 一、本文将分享如何快速搭起基于webpack+vue的前端项目框架,利用vue的自己…

    2025年3月7日 编程技术
    200
  • Webpack优化配置缩小文件搜索范围的介绍

    这篇文章主要介绍了webpack优化-缩小文件搜索范围的相关知识,文中较详细的给大家介绍了可以优化的途径,需要的朋友可以参考下 Webpack 启动后会从配置的 Entry 出发,解析出文件中的导入语句,再递归的解析。 在遇到导入语句时 W…

    编程技术 2025年3月7日
    200
  • webpack原理的深入介绍(附示例)

    本篇文章给大家带来的内容是关于webpack原理的深入介绍(附示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 本文抄自《深入浅出webpack》,建议想学习原理的手打一遍,操作一遍,给别人讲一遍,然后就会了 在阅读前…

    2025年3月7日
    200
  • vue中的webpack用什么安装

    vue中的webpack用node包管理器“npm”或npm镜像“cnpm”来安装。webpack是一个用于现代JavaScript应用程序的静态模块打包工具,是基于node.js开发的,使用时需要有node.js组件支持;需要使用npm或…

    2025年3月7日 编程技术
    200

发表回复

登录后才能评论