webpack执行流程知识点总结

webpack的运行流程

        Webpack 的运行流程是一个串行的过程,从启动到结束会依次执行以下流程:

        在以上过程中,Webpack 会在特定的时间点广播出特定的事件,插件在监听到感兴趣的事件后会执行特定的逻辑,并且插件可以调用 Webpack 提供的 API 改变 Webpack 的运行结果。

webpack 整体流程图

一、初始化流程

  1. 初始化参数:从配置文件和 Shell 语句中读取与合并参数,得出最终的参数。
  2. 安装 webpack-cli;(如果直接使用webpack的核心模块,则不需要安装webpack-cli)
    • npm install webpack webpack-cli -D
    • 这个过程实际上就是用上一步得到的参数初始化Compiler对象,并加载所有配置的插件,通过执行Compiler对象的run方法开始执行编译过程。
  3. 执行 webpack-cli,并将参数传入,运行 webpack。(如果直接使用webpack的核心模块,则需要手动编写代码来调用 webpack 的 API)
    • npx webpack-cli <command> <arguments>
    • 这个过程实际上是调用webpack-cli的主入口文件bin/webpack.js。在执行时,会将命令行参数传递给webpack()函数,这个函数是webpack的主要入口点。然后,webpack内部会解析这些参数并按照webpack的配置来执行编译过程。而entry 属性用于指定入口文件或入口文件的数组。这些入口文件是 webpack 编译的起点,它会根据配置中的entry找出所有的入口文件。

webpack的配置项

        Webpack的配置文件的作用是激活Webpack的加载项和插件,通过定义和配置这些加载项和插件,我们可以更好地处理前端资源,优化构建过程,提高构建效率和应用程序的性能。

  • 该配置文件默认为webpack.config.js,通常位于项目的根目录下;
  • 也可以通过命令的形式指定其他的配置文件,例如,在执行Webpack命令时,可以通过--config参数来指定配置文件的路径。
webpack --config myconfig.js
// 在这个例子中,Webpack将使用myconfig.js作为配置文件,而不是默认的webpack.config.js。

        当Webpack启动时,它会读取webpack.config.js文件,并将配置项拷贝到options对象中,并加载用户配置的插件,并将它们添加到构建过程中。通过这些配置项和插件,我们可以定制Webpack的行为,以满足项目的特定需求,更好地处理前端资源、优化构建过程和提高构建效率。

Shell脚本参数

shell 脚本参数一般通过 shell 直接输入或者通过 npm script 输入。

如上图所示,

①通过 shell 执行 npm run start 后,会自动执行 webpack --config webpack.config.js ,

②通过 shell 的 config 参数将配置文件webpack.config.js传入 webpack 中。

Shell 与 webpack.config 解析

        每次在命令行中输入webpack后,操作系统会查找(当前目录下)./node_modules/.bin/webpack并执行该文件作为shell脚本,以便启动Webpack构建过程。在脚本中,它可能使用了一些命令行参数来进一步定制构建过程。

这个脚本通常会包含以下逻辑:

①调用 ./node_modules/webpack/bin/webpack.js :这是Webpack核心的入口点,它实际上执行了构建过程。

②追加任何附加的命令行参数:例如,-p和-w是两个常见的命令行参数。

  • -p:这个参数通常用于生产环境构建,它会启用一些优化和压缩功能,使构建结果更小、更快。
  • -w:这个参数用于启用Webpack的“watch mode”,使得Webpack在源文件发生变化时自动重新构建,而不是在每次构建时都重新读取整个项目。

这样做是为了提供一种灵活的方式来启动Webpack,并允许用户通过命令行参数来定制构建过程。

(下图中, webpack.js 是 webpack 的启动文件,而 $@ 是后缀参数)

        在命令行中运行Webpack时,可以通过--参数 来传递一些指令给Webpack。这些指令会被解析为一个对象,并与Webpack的配置对象合并。合并后的配置对象会被传递给Webpack的插件和加载器,以便它们根据这些配置来执行相应的任务。

【CLI】optimist

        在 webpack.js 这个文件中 webpack 通过 optimist 将用户配置的 webpack.config.js 和 shell 脚本传过来的参数整合成 options 对象传到了下一个流程的控制对象中。这个 options 对象包含了所有用于控制 Webpack 构建过程的配置和参数。

        为了整合 webpack.config.js 中的配置和命令行参数,Webpack 使用了一个叫做 optimist 的库。optimist 是一个用于解析命令行参数的库,可以将命令行参数解析为 JavaScript 对象,这样它们就可以与 webpack.config.js 中的配置进行合并。它可以帮助 Webpack 将用户通过命令行传递的参数与配置文件中的选项整合在一起,形成一个完整的 options 对象。这个 options 对象会被传递给后续的插件或加载器,以便它们可以根据这些配置来执行相应的任务。

config 合并与插件plugins加载
  • 在加载插件之前,webpack将webpack.config.js中的各个配置项拷贝到options对象中,并加载用户配置在webpack.config.js的plugins。
    • 插件的默认触发阶段主要是在 Webpack 构建过程中的初始化(init)和完成(done) 阶段。
      • 初始化(init)阶段是在Webpack启动时触发的,此时Webpack正在进行一些基础的配置和初始化工作。在这个阶段,插件可以注册事件监听器,以便在后续的构建过程中接收通知。
      • 完成(done)阶段是在Webpack构建完成之后触发的,此时所有的模块都已经打包完成,Webpack正在清理临时文件并关闭相关资源。在这个阶段,插件可以执行一些收尾工作,例如输出日志、清理临时文件等。
  • 接着 optimist.argv 会被传入到./node_modules/webpack/bin/convert-argv.js 中,通过判断 argv 中参数的值决定是否去加载对应插件。
ifBooleanArg("hot", function() {
  ensureArray(options, "plugins");
  var HotModuleReplacementPlugin = require("../lib/HotModuleReplacementPlugin");
  options.plugins.push(new HotModuleReplacementPlugin());
});
...
return options;
//这段代码的目的是检查是否存在一个名为"hot"的参数,
//如果存在并且为真,那么它会确保options对象有一个插件数组,并向该数组中添加一个热模块替换插件的实例。
//最后,返回修改后的options对象。
  • options 作为最后返回结果,包含了之后构建阶段所需的重要信息。
{ 
  entry: {},//入口配置
  output: {}, //输出配置
  plugins: [], //插件集合(配置文件 + shell指令) 
  module: { loaders: [ [Object] ] }, //模块配置
  context: //工程路径
  ... 
}
  • 这和 webpack.config.js 的配置非常相似,只是多了一些经 shell 传入的插件对象。插件对象一初始化完毕, options 也就传入到了下个流程中。
var webpack = require("../lib/webpack.js");
var compiler = webpack(options);
//options是一个对象,包含了Webpack的配置信息,这些配置可以来自文件、环境变量、命令行参数等。
【手动】yargs

        在Webpack中直接获取Shell脚本参数并不直接可能,因为Webpack和Shell脚本在运行时是相互独立的。Webpack主要处理JavaScript和相关资源文件的打包,而Shell脚本则用于执行操作系统命令和操作。只有使用CLI进行构建才能自动获取到Shell里输入的参数;如果是不依赖CLI,手动构建的,那么就需要自己写个脚本去读取和处理Shell 的参数。

        如果想要在Webpack中获取Shell脚本参数,可以使用Node.js的命令行参数解析库,例如yargs或commander。这些库可以帮助你解析命令行参数,并在Webpack的配置文件中使用这些参数。

        yargs 是一个非常有用的 Node.js 库,用于解析命令行参数,可以轻松地定义和解析命令行选项、参数和子命令,并将它们转换为 JavaScript 对象,这个对象可以作为参数传递给其他函数或流程,使得命令行参数的处理更加灵活和方便。

        假设你有一个 webpack.config.js 文件,其中定义了一些 webpack 配置选项,还有一些 shell 脚本参数,你想将它们整合到一个 options 对象中,然后将这个对象传递给下一个流程。下面是一个简单的示例,展示了如何使用 yargs 来实现这个目标:

  • 首先,需要确保已经安装了yargs库:
npm install yargs
  • 创建一个 cli.js 文件,其中包含 yargs 的配置和命令行参数的处理逻辑:
// 引入 yargs 模块,它是命令行参数解析库  
const yargs = require('yargs/yargs');  
// 引入 yargs 的 helpers,它提供了一些辅助函数,如 hideBin,用于隐藏 yargs 的内部命令  
const { hideBin } = require('yargs/helpers');  
// 引入 path 模块,用于处理文件和目录的路径  
const { join } = require('path');  
// 引入 webpack.config.js,假设它包含了一些 webpack 的配置选项  
const webpackConfig = require('./webpack.config.js');  
  
// 使用 yargs 创建一个命令行参数解析器,并隐藏 yargs 的内部命令  
const argv = yargs(hideBin(process.argv))  
  // 定义一个命令行参数 'port',它的别名是 'p',类型是 'number',描述是 '指定服务器监听的端口'  
  .option('port', {  
    alias: 'p',  
    type: 'number',  
    description: '指定服务器监听的端口',  
  })  
  // 定义一个命令行参数 'env',它的别名是 'e',类型是 'string',描述是 '指定环境变量'  
  .option('env', {  
    alias: 'e',  
    type: 'string',  
    description: '指定环境变量',  
  })  
  // 添加一个帮助选项,当用户在命令行中输入 --help 时,会显示帮助信息  
  .help()  
  // 解析命令行参数,将解析结果存储在 argv 对象中  
  .argv;  
  
// 将 webpack.config.js 中的配置和命令行参数合并到一个 options 对象中  
const options = { ...webpackConfig, port: argv.port, env: argv.env };  
// 打印 options 对象,以供调试或输出给下一个流程使用  
console.log(options);

在这个示例中,我们定义了两个命令行参数:port 和 env。然后,我们将 webpack.config.js 中的配置和这两个命令行参数合并到一个 options 对象中。最后,我们打印出这个 options 对象。

  • 运行这个脚本:
node cli.js --port 3000 --env production

以下是另外一个使用yargs库的示例(必须传要求的参数):

  • 首先,你需要安装yargs库:
npm install yargs
  • 在Webpack配置文件中,引入yargs库并解析命令行参数:
// 导入yargs库  
const yargs = require('yargs/yargs');  
// 导入hideBin帮助函数,它用于隐藏命令行中的二进制部分  
const { hideBin } = require('yargs/helpers');  
  
// 使用yargs创建一个命令行参数对象  
const argv = yargs(hideBin(process.argv))  
  // 定义一个名为'input'的命令行参数,用户可以通过'-i'或'--input'来指定  
  .option('input', {    
    alias: 'i',    
    type: 'string',  // 参数类型为字符串  
    description: 'Input file path',  // 描述该参数的作用,即输入文件路径  
  })    
  // 定义一个名为'output'的命令行参数  
  .option('output', {    
    alias: 'o',    
    type: 'string',  // 参数类型为字符串  
    description: 'Output file path',  // 描述该参数的作用,即输出文件路径  
  })    
  // 如果用户没有提供'input'和'output'这两个参数,则显示错误信息  
  .demandOption(['input', 'output'], 'You must provide both input and output file paths')    
  // 显示帮助信息,当用户在命令行中输入'-h'或'--help'时显示  
  .help()    
  // 获取解析后的命令行参数对象  
  .argv;
  • 在Webpack配置中使用解析后的参数:
module.exports = {  
  // ...其他配置项...  
  // 入口文件路径,从命令行参数中获取  
  entry: argv.input,  
  // 输出配置  
  output: {  
    // 输出文件的路径,相对于当前目录,使用命令行参数指定  
    path: path.resolve(__dirname, argv.output),  
    // 输出文件的名称  
    filename: 'bundle.js',  
  },  
};
  • 这样,当在命令行中运行Webpack时,可以通过指定命令行参数来传递输入和输出文件路径。例如:
npx webpack --input=src/index.js --output=dist/bundle.js

        在这个例子中,--input和--output参数分别指定了输入和输出文件路径,你可以在Webpack配置中使用argv.input和argv.output访问这些参数,根据传递的参数来指定打包的入口和输出文件的位置。

执行之后,Shell命令行中传入的参数值将被合并到配置对象中,并覆盖默认值或先前指定的值。

下面是大佬的代码,码住:

// 定义一个异步函数,接受一个可选参数platform,默认为空字符串  
async function runWebpackBuild(platform = '' ) {  
  // 返回一个新的Promise对象  
  return new Promise(async (resolve, reject) => {  
    try {  
      // 根据传入的platform参数,构建webpack构建脚本的路径  
      const builderPath = path.resolve(__dirname, `./webpack.${platform}.build.js`)  
      // 构建并执行Node.js命令,用于运行webpack构建脚本  
      const nodeCmd = `node ${builderPath} --lang=${argvs.lang || 'my-en'} --secondLang=${argvs.secondLang || ''} --env=${argvs.env} --staticSource=${argvs.staticSource}`}  
      // 这行代码被注释掉了,如果取消注释,它将打印出要执行的Node.js命令  
      // consola.info(`>>>> 执行命令: ${nodeCmd}` )  
      // 执行Node.js命令,等待命令执行完成  
      await execCmd( nodeCmd )  
      // 如果命令执行成功,则返回一个解析的Promise,表示构建成功  
      return resolve(true)  
    } catch (error) {  
      // 如果命令执行出错,则返回一个拒绝的Promise,表示构建失败  
      return reject(error)  
    }  
  })  
}

二、编译与构建流程

        在加载配置文件和shell后缀参数申明的插件,并传入构建信息options对象后,开始整个webpack打包最漫长的一步。而这个时候,真正的webpack对象才刚被初始化,具体的初始化逻辑在 ./node_modules/webpack/lib/webpack.js中,如下所示:

//Compiler 类继承了 Tapable。Compiler对象定义在./node_modules/webpack/lib/Compiler.js
//Tapable 是一个提供钩子(hooks)功能的基类,
//这些钩子允许我们在特定的事件点上执行自定义逻辑。
class Compiler extends Tapable {
    constructor(context) {
        super();
        this.hooks = {
            //生命周期钩子
            beforeCompile: new AsyncSeriesHook(["params"]),
            compile: new SyncHook(["params"]),
            afterCompile: new AsyncSeriesHook(["compilation"]),
            make: new AsyncParallelHook(["compilation"]),
            entryOption: new SyncBailHook(["context", "entry"])
            // 定义了很多不同类型的钩子
        };
        // ...
    }
}

//这是一个Webpack的外部API函数,它接受一个配置对象options作为参数。
//在函数内部创建了一个新的Compiler实例。这意味着使用Webpack时,实际上是在使用这个Compiler类来处理和打包资源。
function webpack(options) {
  var compiler = new Compiler();
  ...// 检查options,若watch字段为true,则开启watch线程
  return compiler;
}
...
在 lib/webpack.js 文件中,Compiler 对象的初始化逻辑主要包括以下几个步骤:
创建 Compiler 实例:通过调用 new Compiler() 创建一个新的 Compiler 实例。
配置 Compiler:根据传入的配置对象(如 webpack.config.js 中定义的配置),对 Compiler 进行相应的配置,例如设置输出目录、加载器等。
注册插件通过调用compiler.hooks上的方法,注册各种插件。这些插件可以在构建过程中执行各种自定义操作,如资源优化、代码分割等。
初始化其他组件:根据配置 初始化其他与构建相关的组件,如Compilation、ResolverFactory等。
返回 Compiler 实例:将初始化完成的 Compiler 实例返回,以便在构建过程中使用。

        当 Webpack 初始化完成后,Compiler 对象会存储所有的配置和组件,但实际的构建过程还没有开始,要开始构建过程,需要调用 Compiler 对象的 run 方法。调用 Compiler 对象的 run 方法真正启动webpack构建流程,它是启动构建过程的最后一步,也是实际生成打包文件的开始,在 run 方法中,Webpack 会执行一系列的步骤来处理资源、打包代码、执行插件等,这些步骤会根据配置和插件的逻辑来执行,最终生成打包后的文件。

        webpack 的实际入口是 Compiler 中的 run 方法,run 一旦执行后,就开始了编译和构建流程 ,其中有几个比较关键的 webpack 事件节点。

  • compile 开始编译
    • 目的:这是整个编译过程的开始,意味着开始将源代码转换为可执行代码或目标代码。
    • 操作:读取源代码文件,并开始进行初步的解析和处理。
  • make 从入口点分析模块及其依赖的模块,创建这些模块对象
    • 目的:确定项目的依赖关系,并创建模块对象以供后续处理。
    • 操作:工具(如Webpack)会分析源代码中的import或require语句,分析项目的依赖关系,并确定哪些模块需要被编译。为每个模块创建一个对象,这个对象包含了该模块的所有相关信息。
  • build-module 构建模块
    • 目的:对每个模块的源代码进行处理和转换。
    • 操作:这一步包括将模块的源代码进行转换、优化、合并等操作。例如,Babel会在此阶段将ES6+的语法转换为ES5语法。
  • after-compile 完成构建
    • 目的:在所有模块都构建完成后执行一些全局的操作。
    • 操作:例如,可能会进行全局的代码优化、清理临时文件等。
  • seal 封装构建结果
    • 目的:将构建的结果进行封装,以便于部署和分发。
    • 操作:可能会将所有的模块和资源打包到一个或多个文件中,并进行压缩、优化等操作。
  • emit 把各个chunk输出到结果文件
    • 目的:将各个模块或chunk输出到结果文件。
    • 操作:工具会根据其内部的依赖关系,将每个chunk(例如,由多个模块组成的代码块)输出到指定的结果文件。
  • after-emit 完成输出
    • 目的:在所有chunks都输出到结果文件后执行一些后续操作。
    • 操作:例如,可能会复制生成的资源到输出目录、执行一些清理任务等。

核心对象 Compilation

        在 Webpack 的构建过程中,当调用compiler.run方法后,会触发一系列的编译阶段。其中,compile阶段是这些阶段中的第一个。

compiler.run 后首先会触发 compile ,这一步会构建出 Compilation 对象。

这个Compilation对象是编译阶段的主要执行者,有两个作用,

  • 一、负责组织整个打包过程,包含了每个构建环节及输出环节所对应的方法,可以从图中看到比较关键的步骤,如 addEntry() , _addModuleChain() ,buildModule() , seal() , createChunkAssets() (在每一个节点都会触发 webpack 事件去调用各插件)。
  • 二、该对象内部存放着所有module,chunk,生成的 asset 以及用来生成最后打包文件的template的信息。

        Compilation对象是Webpack内部工作机制的重要部分,它具有大量的钩子(hooks),这些钩子可以被插件用来插入自定义的逻辑。当检测到一个文件变化时,Webpack会创建一个新的Compilation对象,这样它就可以重新编译并加载进内存。在编译阶段,模块会被加载、封存、优化、分块、哈希和重新创建。这些过程都与Compilation对象紧密相关。

编译与构建主流程

        在创建module之前,Compiler会触发make,并调用Compilation.addEntry方法,通过options对象的entry字段找到我们的入口js文件。

之后,在addEntry中调用私有方法_addModuleChain,

这个_addModuleChain方法主要做了两件事情。

一是根据模块的类型获取对应的模块工厂并创建模块,

二是构建模块。

获取对应的模块工厂并创建模块

        _addModuleChain 是 Webpack 的内部方法,用于处理模块链,能根据给定的依赖项创建新的模块实例。这个方法主要的工作是根据给定的 module 和 context 对象,以及一些其他参数,来创建一个新的模块。具体来说,这个方法会根据模块的类型(例如,JavaScript、CSS、图片等)来获取对应的模块工厂并创建模块。

_addModuleChain(context, dependency, onModule, callback) {
   ...
   // 根据依赖查找对应的工厂函数
   const Dep = /** @type {DepConstructor} */ (dependency.constructor);
   const moduleFactory = this.dependencyFactories.get(Dep);
   
   // 调用工厂函数moduleFactory的create来生成一个空的NormalModule对象,NormalModul是Webpack中表示一个模块的类。 
   moduleFactory.create({
       dependencies: [dependency]
       ...
   }, (err, module) => {
       ...
       //定义一个afterBuild函数,这个函数在模块编译完成后会被调用。 
       const afterBuild = () => {
        this.processModuleDependencies(module, err => {
          if (err) return callback(err);//处理过程中发生错误,则通过回调函数返回错误信息 
           callback(null, module); //处理成功,则通过回调函数返回新创建的模块
         });
    };
       //开始模块编译。这里会调用buildModule方法来进行实际的编译操作。  
       this.buildModule(module, false, null, null, err => {
           ...
           afterBuild();//当模块编译完成后,调用之前定义的 afterBuild函数。 
       })
   })
}// _addModuleChain 方法结束
_addModuleChain 是 Webpack 的内部方法,用于处理模块链。以下是该方法的主要过程:
1. 接收参数:该方法接收四个参数,包括上下文对象 context、依赖项 dependency、回调函数 onModule 和回调函数 callback。
2. 获取模块工厂:根据依赖项的构造类型,从 dependencyFactories 集合中获取对应的模块工厂。
3. 创建新模块:使用获取到的模块工厂,通过 create 方法创建一个新的模块实例。
4. 添加模块到编译队列:将新创建的模块加入到编译对象的 modules 数组中,以便在后续的打包过程中使用该模块。
5. 处理模块依赖:开始处理新创建模块的依赖项,递归地将它们也加入到编译队列中。
6. 编译模块:当所有依赖项都处理完毕后,开始编译当前模块。Webpack 会调用 buildModule 方法来执行实际的编译操作,包括解析模块代码、生成依赖图等。
7. 处理编译结果:当模块编译完成后,Webpack 会进行一些后续处理,例如将模块添加到编译对象的 modules 数组中,以便在后续的打包过程中使用该模块。
8. 回调函数:当整个过程完成后,通过回调函数返回编译结果或错误信息。
构建模块

而构建模块作为最耗时的一步,又可细化为三步:

1、调用各 loader 处理模块之间的依赖

        webpack提供的一个很大的便利就是能将所有资源都整合成模块,不仅仅是 js文件。因此,需要一些loader ,比如url-loader、jsx-loader、css-loader等,来让我们可以直接在源文件中引用各类资源。

        对每一个require()用对应的loader进行加工:webpack调用doBuild(),对每一个require()用对应的loader进行加工,最后生成一个js module。当webpack遇到require()或import语句时,它会使用对应的加载器(loader)处理这些依赖项。加载器可以将源文件转换成Webpack能够理解和处理的模块。然后,Webpack会调用acorn来解析这些经过加载器处理的源文件,生成抽象语法树(AST)。

// 定义在Compilation对象上的_addModuleChain方法,该方法用于处理模块链。  
// 接收四个参数:context(上下文)、dependency(依赖项)、onModule(处理模块的回调函数)和callback(回调函数)。  
Compilation.prototype._addModuleChain = function process(context, dependency, onModule, callback) {
  // 如果Compilation对象启用了性能分析,则记录当前时间,用于后续计算该方法的执行时间。 
  var start = this.profile && +new Date();
  ...
  // 根据模块的类型获取对应的模块工厂并创建模块
  // 根据传入的依赖项的构造函数,从dependencyFactories集合中获取对应的模块工厂。  
  // 这个集合中存储了各种类型的模块工厂,用于创建不同类型的模块。
  var moduleFactory = this.dependencyFactories.get(dependency.constructor);
  ...
  moduleFactory.create(context, dependency, function(err, module) {
    var result = this.addModule(module); // 将新创建的模块添加到Compilation对象的modules数组中。
    ...
    this.buildModule(module, function(err) { // 调用buildModule方法来构建模块。
      ...
      // 构建模块,添加依赖模块
    }.bind(this)); // 由于使用了bind(this)来绑定回调函数的执行上下文为Compilation对象,所以后续操作可以在Compilation对象的上下文中进行。
  }.bind(this));
};

2、调用 acorn 解析经 loader 处理后的源文件,生成抽象语法树 AST

        Acorn是一个轻量级、流式的JavaScript解析器,可以生成抽象语法树(AST)。通过Acorn解析源文件后,可以获得源文件的语法结构,以便进行进一步的分析、转换或打包。虽然加载器在处理源文件时起到重要作用,但AST的生成是由Acorn解析器完成的。

// 定义Parser对象的parse方法,该方法用于解析源代码并返回抽象语法树(AST)。  
// 它接收两个参数:source(待解析的源代码)和initialState(初始状态)。 
 Parser.prototype.parse = function parse(source, initialState) {
  var ast;
  if (!ast) {
    // acorn以es6的语法进行解析
    ast = acorn.parse(source, {
      ranges: true, // 解析时保留每个语法元素的原始位置范围信息。  
      locations: true, // 解析时保留每个语法元素的原始位置信息。  
      ecmaVersion: 6, // 使用ECMAScript 6(即ES6)的语法版本进行解析。  
      sourceType: "module" // 将源代码视为模块,解析为模块的抽象语法树(AST)。  
    });
  }
  ...
};

3、遍历AST,构建该模块所依赖的模块。

对于当前模块,或许存在着多个依赖模块。当前模块会开辟一个依赖模块的数组,在遍历AST时,将require()中的模块通过addDependency()添加到数组中。当前模块构建完成后,webpack调用processModuleDependencies开始递归处理依赖的module,接着就会重复之前的构建步骤。

// 定义Compilation对象的addModuleDependencies方法,用于向给定的模块添加依赖。  
// 它接收五个参数:module(要添加依赖的模块)、dependencies(依赖数组)、bail(是否立即停止编译)、cacheGroup(缓存组)和recursive(是否递归添加依赖)。  
// 最后一个参数callback是回调函数,用于在添加依赖完成时进行回调。
 Compilation.prototype.addModuleDependencies = function(module, dependencies, bail, cacheGroup, recursive, callback) {
  // 根据依赖数组(dependencies)创建依赖模块对象
  var factories = []; // 用于存储依赖模块的工厂对象
  for (var i = 0; i < dependencies.length; i++) { // 为每个依赖创建一个工厂对象并存储到factories数组中。
    var factory = _this.dependencyFactories.get(dependencies[i][0].constructor); // 获取对应的模块工厂
    factories[i] = [factory, dependencies[i]]; // 将工厂对象和依赖项一起存储到factories数组中。  
  }
  ...
  // 与当前模块构建步骤相同
}
构建细节

        module是webpack构建的核心实体,也是所有module的父类,它有几种不同子类:NormalModule,MultiModule,ContextModule,DelegatedModule 等。但这些核心实体都是在构建中都会去调用对应方法,也就是模块构建函数build()。

它主要完成了以下几件事情:
初始化模块信息:设置模块的构建时间戳,并标记模块为已构建。
检查是否需要跳过解析:如果提供了module.noParse选项,并且是字符串或正则表达式数组,代码会检查当前模块的请求是否匹配其中的某个模式。如果匹配,则直接返回回调,不进行后续解析操作。
解析源代码:使用acorn库解析模块的源代码,生成抽象语法树(AST)。
错误处理:如果在解析过程中发生错误,代码会捕获异常,获取源代码内容,清空源代码引用,并抛出一个包含错误信息和源代码内容的ModuleParseError异常。
返回回调:如果一切正常,或者在跳过解析的情况下,函数会返回回调。

        对于每一个module,它都会有这样一个构建方法build()。当然,它还包括了从构建到输出的一系列的有关module生命周期的函数,我们通过module父类类图其子类类图(这里以NormalModule为例)来观察其真实形态:

        可以看到,无论是构建流程,处理依赖流程,还是后面的封装流程都是与module密切相关的。

三、打包输出流程

        在所有模块及其依赖模块编译(build)完成后,webpack会触发一个名为seal的生命周期事件,这个事件标志着构建过程的结束,并且webpack将开始对构建后的结果进行封装。在seal事件触发后,webpack会逐个对每个module和chunk进行整理,这个阶段的目标是生成编译后的源码,进行合并、拆分以及生成hash。

当seal事件被触发时,Webpack会执行以下操作:
1. 调用插件的seal方法:Webpack会通知所有已注册的插件,构建过程已经完成,并可以开始执行自定义的seal方法。这允许插件对构建后的结果进行自定义处理或添加额外的元数据。
2. 整理模块和chunks:Webpack会对所有模块和chunks进行整理,确保它们按照正确的顺序排列。这包括根据模块的依赖关系进行排序,以及合并和拆分代码块。
3. 生成最终的assets:在整理完成后,Webpack会生成最终的assets,包括JavaScript、CSS、图片等文件。这些assets是构建过程的输出结果,可以直接部署到生产环境中使用。
4. 触发其他插件事件:在seal事件被触发后,Webpack可能还会触发其他插件事件,例如optimize-tree等。这些事件允许插件执行进一步优化或其他自定义操作。
简而言之,seal事件是Webpack构建过程中的一个重要环节,它标志着构建过程的结束,并允许插件对构建后的结果进行封装和自定义处理。

        同时这是我们在开发时进行代码优化和功能添加的关键环节。例如,可以利用seal事件对代码进行压缩、混淆或优化,以减少文件大小和提高性能。还可以使用插件来添加元数据、生成文档或执行其他与构建结果相关的任务。可以触发其他插件事件,如optimize-tree,以进一步优化模块和代码树的结构。这些优化可以提高应用程序的性能、减少加载时间并改善用户体验。

// 定义Compilation对象的seal方法,该方法接受一个回调函数作为参数  
Compilation.prototype.seal = function seal(callback) {
  this.applyPlugins("seal");//调用并触发所有已注册的插件的seal事件。这允许插件执行一些自定义逻辑或操作  
  this.preparedChunks.sort(function(a, b) {//对已准备的chunks进行排序。排序的依据是chunk的名称。 
    if (a.name < b.name) {
      return -1;//如果名称a小于名称b,则返回-1;
    }
    if (a.name > b.name) {
      return 1;//如果名称a大于名称b,则返回1;
    }
    return 0;//如果两者相等,则返回0 
  });
  // 遍历已排序的chunks,并对每个chunk执行以下操作:
  this.preparedChunks.forEach(function(preparedChunk) {
    var module = preparedChunk.module;//获取与当前chunk关联的模块 
    var chunk = this.addChunk(preparedChunk.name, module);//向当前的chunks集合中添加一个新的chunk,chunk的名称为preparedChunk.name,并且与module关联  
    chunk.initial = chunk.entry = true;//设置新添加的chunk的一些属性:它是一个初始chunk,也是一个入口chunk 
    // 整理每个Module和chunk,每个chunk对应一个输出文件。将module添加到chunk中,并把chunk添加到module中。这表示module和chunk之间存在依赖关系
    chunk.addModule(module);
    module.addChunk(chunk);
  }, this);//使用bind确保回调函数中的this指向Compilation对象实例  
  //异步调用并触发所有已注册的插件的"optimize-tree"事件。这个事件允许插件对chunks和modules进行优化。优化完成后,会执行回调函数。
  this.applyPluginsAsync("optimize-tree", this.chunks, this.modules, function(err) {
    if (err) {
      return callback(err);//如果回调函数中发生错误,则将错误传递给外部的callback函数
    }
    ... // 触发插件的事件
    this.createChunkAssets(); //生成最终assets,可能包括JavaScript、CSS、图片等。生成的assets会被存储在compiler的assets属性中。
    ... // 触发插件的事件
  }.bind(this));// 使用bind确保回调函数中的this指向Compilation对象实例 
};// seal方法结束

生成最终 assets

        在封装过程中,webpack会调用Compilation中的createChunkAssets方法进行打包后代码的生成。createChunkAssets函数通过收集和封装代码片段,将它们转化为可执行的模块,并将这些资源文件存储在compiler.assets对象中,以便后续的输出和引用。createChunkAssets流程如下:

createChunkAssets的流程主要包括以下步骤:
判断是否是入口chunk:createChunkAssets函数首先判断当前处理的chunk是否是入口chunk。如果是入口chunk,则使用MainTemplate进行封装,否则使用ChunkTemplate进行封装。
处理依赖关系:在语法树解析阶段,createChunkAssets函数会收集文件的依赖关系,包括需要插入或替换的部分。
生成代码片段:在generate阶段,createChunkAssets函数会生成各种代码片段,包括需要打包输出的资源。
封装代码:createChunkAssets函数会对生成的代码片段进行收集和封装,将它们封装成可执行的模块。一旦代码片段被封装完毕,它们就会被存储在compiler.assets对象中。

输出最终文件

        在Webpack的打包过程中,createChunkAssets方法执行完毕后,会调用Compiler中的emitAssets()方法。所以最后一步是,webpack调用Compiler中的emitAssets(),将生成的代码输入到output的指定位置,完成最终文件的输出,从而webpack整个打包过程结束。

output对象在Webpack配置中定义了输出目录、文件名等选项,emitAssets()方法会将生成的代码按照output配置的规则输出到相应的目录中。输出的文件可以是JavaScript、CSS、图片等资源文件,以及其他依赖图和其他元数据。

        值得注意的是,若想对结果进行处理,则需要在emitAssets()触发后对自定义插件进行扩展,可以通过编写自定义插件来实现。例如,可以编写一个自定义插件,并在emitAssets()触发后对生成的代码进行进一步处理或修改。在插件的apply方法中,可以访问compiler对象并监听emit事件。当emit事件触发时,可以执行自己的逻辑来处理生成的代码或资源。

具体来说,可以在插件中实现以下步骤:
①访问compiler对象并监听emit事件。
②在emit事件触发时,获取生成的代码或资源。
③对获取的代码或资源进行进一步处理或修改。
④如果你想将修改后的结果输出到文件系统,可以使用compiler.outputFileSystem来写入文件。
⑤完成处理后,你可以选择将修改后的结果重新赋值给compiler.assets对象,以便Webpack将其输出到指定的目录中。

构建方法

使用webpack 可以有两种方式:

①使用 webpack-cli 构建;

②使用 webpack 核心对象构建;

一、使用 webpack-cli 构建

1.安装依赖:在项目根目录下,打开终端并运行以下命令来安装webpack和webpack-cli

npm install --save-dev webpack webpack-cli
// --save-dev 参数的作用是将安装的包作为开发依赖添加到项目的package.json文件中。
// 这意味着这些包仅在开发过程中需要,而不是在生产环境中。

2.创建webpack.config.js文件:在项目根目录下创建一个名为webpack.config.js的文件。这个文件将包含构建配置。

3.配置文件内容:根据你的项目需求,你可以在webpack.config.js中设置各种配置项,例如入口文件、输出文件、加载器、插件等。下面是一个简单的示例:

const path = require('path');  
  
module.exports = {  
  entry: './src/index.js',  
  output: {  
    path: path.resolve(__dirname, 'dist'),  
    filename: 'bundle.js',  
  },  
  module: {  
    rules: [  
      {  
        test: /\.js$/,  
        exclude: /node_modules/,  
        use: 'babel-loader',  
      },  
    ],  
  },  
  plugins: [],  
};

4.运行构建:在你的项目根目录下,运行以下命令来构建项目:

npx webpack --config webpack.config.js

5.启动开发服务器(可选):如果想在开发过程中自动重新构建项目,可以使用webpack-dev-server。

  • 首先,安装它:
npm install --save-dev webpack-dev-server
  • 然后,在package.json中添加一个脚本来启动开发服务器:
"scripts": {  
  "start": "webpack serve --open --config webpack.config.js"  
}
  • 现在,可以通过运行npm start来启动开发服务器,并在浏览器中查看项目。当我们对代码进行更改时,服务器会自动重新构建和刷新浏览器。

二、使用 webpack 核心模块手动构建

        如果不使用webpack CLI,可直接使用webpack的模块化API来手动构建项目。可以通过Node.js的require()函数来手动导入webpack和webpack配置文件,并使用webpack的API进行构建。

1.首先,确保已经安装了webpack和webpack-core。

npm install webpack webpack-core --save-dev
//在Webpack的构建流程中,Webpack-core扮演着重要的角色,
//它负责解析入口文件,递归读取引入文件所依赖的文件内容,
//生成AST语法树,并根据AST语法树生成浏览器能够运行的代码。

2.然后,在需要使用webpack的JavaScript文件中,使用以下代码导入webpack:

const webpack = require('webpack');

3.接下来,导入webpack配置文件。假设webpack配置文件名为webpack.config.js,使用以下代码将其导入:

const config = require('./webpack.config.js');

4.现在,可以使用webpack的API进行构建。

①以下是一个示例,展示了如何使用webpack进行构建:

  • 引入webpack:const webpack = require('webpack');
  • 调用 webpack 并传入配置参数,webpack 将返回一个编译对象;
  • 调用编译对象的 run 方法,run 方法将执行编译。
// 导入webpack模块,这样我们就可以使用webpack的API进行构建  
const webpack = require('webpack');  
//【const path = require('path');】
//【const fs = require('fs');】

// 导入我们的webpack配置文件。这个文件包含了webpack如何处理我们的项目的所有信息  
const config = require('./webpack.config.js');  //*****读取webpack配置文件*****
//【const config = JSON.parse(fs.readFileSync('webpack.config.js', 'utf8'));】 
  
// 使用webpack的API执行构建过程。这里的config是我们之前导入的webpack配置  
const compiler = webpack(config);  //*****创建webpack实例*****
// 运行编译实例。编译实例执行后,会调用回调函数,将错误(如果有的话)和构建统计信息作为参数传入  
compiler.run((err, stats) => {  //*****编译入口文件*****  
  // 当构建过程完成时,回调函数会被调用。如果构建过程中出现错误,err会被赋值。stats对象包含了构建过程中的各种信息  
  if (err || stats.hasErrors()) {  
    // 如果err或stats.hasErrors()返回true,说明构建过程中出现了错误,我们将其打印出来  
    console.error('编译出错:', err);  
  } else {  
    // 如果构建过程没有错误,我们打印一条成功消息  
    console.log('编译完成:', stats.toString({ colors: true }));    
  }  
});

        在这个示例中,我们首先使用require()函数导入webpack和path模块。然后,我们使用fs模块读取webpack.config.js配置文件,并将其解析为JSON对象。接下来,我们创建一个webpack实例,并将配置文件传递给它。最后,我们调用compiler.run()方法来执行编译过程。

  • 使用compiler.run()可启动编译过程,使用compiler.watch()可监视文件变化并自动编译。

②还可以用另一种写法:

// 导入webpack模块,这样我们就可以使用webpack的API进行构建  
const webpack = require('webpack');  
  
// 创建一个Webpack配置对象。这个对象包含了webpack如何处理我们的项目的所有信息  
const config = {  
  // 指定入口文件的路径。这里假设入口文件为./src/index.js  
  entry: './src/index.js',   
  // 配置输出目录的路径。这里假设输出目录为项目根目录下的dist文件夹  
  output: {  
    path: path.resolve(__dirname, 'dist'),   
    // 指定输出文件的名称。这里假设输出文件名为bundle.js  
    filename: 'bundle.js'   
  }  
};  
  
// 使用webpack的API执行构建过程。这里的config是我们之前创建的Webpack配置对象  
webpack(config, (err, stats) => {  
  // 当构建过程完成时,回调函数会被调用。如果构建过程中出现错误,err会被赋值。stats对象包含了构建过程中的各种信息  
  if (err || stats.hasErrors()) {  
    // 如果err或stats.hasErrors()返回true,说明构建过程中出现了错误,我们将其打印出来  
    console.error(err);  
  } else {  
    // 如果构建过程没有错误,我们打印一条成功消息  
    console.log('Webpack build completed successfully.');  
  }  
});

        在上面的示例中,webpack(config, callback)函数用于执行构建。config参数是要使用的webpack配置对象,callback参数是一个回调函数,在构建完成后被调用。如果构建过程中出现错误或警告,它们将作为回调函数的参数传递给err和stats。我们可以根据需要进行错误处理或输出构建结果。

5.运行Webpack: 在命令行终端中,运行以下命令来执行Webpack:

npx webpack --config webpack.config.js
  • 可以在项目的根目录下创建一个package.json文件(如果还没有的话),并添加以下内容:
{  
  "name": "my-webpack-project",  
  "version": "1.0.0",  
  "scripts": {  
    "build": "webpack --config webpack.config.js" // 添加一个构建脚本  
  },  
  "devDependencies": {  
    "webpack": "^5.0.0", // 指定Webpack的版本号,根据实际情况进行调整  
  }  
}
  • 在命令行终端中,运行以下命令来构建项目
npm run build // 使用npm命令构建项目

6.启动开发服务器(可选):

  • 首先,确保你已经安装了 webpack 和 webpack-dev-server。
npm install --save-dev webpack webpack-dev-server
  • 在项目 webpack.config.js 文件中添加以下内容
const path = require('path');  
  
module.exports = {  
  entry: './src/index.js', // 指定入口文件  
  output: {  
    path: path.resolve(__dirname, 'dist'), // 指定输出目录  
    filename: 'bundle.js', // 指定输出文件名  
  },  
  devServer: {  
    contentBase: './dist', // 指定服务器响应的静态资源目录  
    port: 3000, // 指定服务器监听的端口号  
  },  
};

  • 可以在项目的根目录下创建一个 package.json 文件(如果还没有的话),并添加以下内容
{  
  "name": "my-webpack-project",  
  "version": "1.0.0",  
  "scripts": {  
    "start": "webpack serve --open" // 添加一个启动脚本  
  },  
  "devDependencies": {  
    "webpack": "^5.0.0", // 指定 webpack 的版本号,根据实际情况进行调整  
    "webpack-dev-server": "^3.11.0" // 指定 webpack-dev-server 的版本号,根据实际情况进行调整  
  }  
}

  • 打开终端,进入项目根目录,并运行以下命令启动开发服务器
npm start // 使用 npm 命令启动服务器

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/320939.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ArrayBlockingQueue的使用

异步日志打印模型概述 在高并发、高流量并且响应时间要求比较小的系统中同步打印日志已经满足不了需求了&#xff0c;这是因为打印日志本身是需要写磁盘的&#xff0c;写磁盘的操作会暂时阻塞调用打印日志的业务线程&#xff0c;这会造成调用线程的rt增加。 如图所示为同步日…

docker部署ng实现反向代理

场景 按规定尽可能减少开放到外网的端口&#xff0c;所以需要将多个服务部署到一个ip一个端口上。 方案 使用ng实现请求转发。根据http请求中的host与ng配置文件中的server_name匹配&#xff0c;转发到对应的机器上。 在docker上部署三个容器&#xff0c;每个容器中启动一个…

JavaScript系列——Promise

文章目录 概要Promise三种状态状态改变Promise链式调用Promise处理并发promise.all()promise.allSettled&#xff08;&#xff09;Promise.any()promise.race() 小结 概要 Promise中文翻译过来就是承诺、预示、有可能的意思。 在JavaScript里面&#xff0c;Promise 是一个对象…

2023年山东省职业院校技能大赛高职组信息安全管理与评估 理论题(正式赛)

2023年山东省职业院校技能大赛高职组信息安全管理与评估 理论题 理论技能与职业素养&#xff08;100分&#xff09; 2023年山东省职业院校技能大赛高职组信息安全管理与评估 理论题 【注意事项】 Geek极安云科专注技能竞赛技术提升&#xff0c;基于各大赛项提供全面的系统性…

【JAVA】哪些集合类是线程安全的

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;JAVA ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 前言 正文 Vector&#xff1a; HashTable&#xff1a; Collections.synchronizedList()、Collections.synchronizedSet()、Collections.syn…

网络地图服务(WMS)详解

文章目录 1.概述2.GetCapabilities3.GetMap4.GetFeatureInfo 阅读本文之前可参考前文&#xff1a;《地图服务器GeoServer的安装与配置》与《GeoServer发布地图服务&#xff08;WMS、WFS&#xff09;》。 1.概述 经过前文的介绍&#xff0c;相信我们对WMS/WFS服务已经有了一个非…

收银系统源码收银系统OEM定制开发(收银POS+线上商城+ERP+营销插件+聚合支付)

源码&#xff1a;零售行业线下线上一体化收银系统 1.开发语言 核心开发语言: php、HTML5、JavaPHP开发环境&#xff1a;php7.3安卓端收银、助手: 原生的JavaPC收银端: HTML5vuecssjsnwjs微信小程序助手: 原生小程序开发商城: uniapp后合管理网站: HTML5vueelement-uicssjs 2…

基于Java SSM框架实现在线作业管理系统项目【项目源码】计算机毕业设计

基于java的SSM框架实现在线作业管理系统演示 JSP技术 JSP技术本身是一种脚本语言&#xff0c;但它的功能是十分强大的&#xff0c;因为它可以使用所有的JAVA类。当它与JavaBeans 类进行结合时&#xff0c;它可以使显示逻辑和内容分开&#xff0c;这就极大的方便了运动员的需求…

day01

文章目录 创建Vue实例插值表达式响应式数据常见标签v-htmlv-show v-ifv-if v-else-if v-elsev-onv-bindv-forv-model 综合案例 创建Vue实例 <!--创建Vue实例&#xff0c;初始化渲染1. 准备容器2. 引包&#xff08;官网&#xff09; — 开发版本/生产版本3. 创建Vue实例 ne…

配置DNS

vim /etc/named.conf vim /etc/named.rfc1912.zones cp named.localhost ./kgc.com.zone -p vim kgc.com.zone 设置备用dns服务器 修改主配置文件&#xff0c;并自动同步到从服务器

Zung氏焦虑症测试SAS

SAS被称为焦虑自评量表&#xff0c;是一种用来测量焦虑症状程度以及观察治疗过程中变化情况的心理量表。主要用于评估心理状态&#xff0c;辅助参考数据&#xff0c;也是焦虑评定的标准。焦虑自评量表系是由William W.K. Zung编制的&#xff0c;该量表已成为心理咨询师、心理医…

COBOL语言 :一种主要专注于解决业务问题的编程语言

译文&#xff1a; 什么是COBOL? COBOL是一种主要专注于解决业务问题的编程语言。COBOL的完整形式是面向业务的通用语言。它主要用于公司和政府的商业、金融和行政系统。这种语言也被用来解决许多数据处理问题。 它是由CODASYL(数据系统语言会议)开发的。它被用作大型机中的一…

解决报错:AttributeError: module ‘torch‘ has no attribute ‘_six‘

报错代码&#xff1a; if torch._six.PY3:import importlibimport importlib.utilimport sys这里的意思是判断你的python版本是否是python3&#xff0c;如果是PY37&#xff0c;就是判断你的python版本是不是python3.7&#xff0c;源码&#xff1a; PY37 sys.version_info[0]…

Keil5如何生成反汇编文件

Keil5如何生成反汇编文件 在Keil5界面下点击选项&#xff0c;选择“User”&#xff0c;勾选“After Build/Rebuild”中“RUN #1”&#xff0c;复制fromelf --text -a -c --outputxxx.dis xxx.axf 在Linker栏中找到“Linker Control string”里最后-o后的.axf文件&#xff0c;将…

街机模拟游戏逆向工程(HACKROM)教程:[1]数据的存储与读取

简介 在计算机中&#xff0c;数据存储的介质一直在变化&#xff0c;从最早的穿孔纸带&#xff0c;到现在的固态硬盘。但存储的原理是一直没有变化的&#xff0c;在计算机中&#xff0c;我们所存储的数据&#xff0c;一直都是以二进制的形式被存储存在不同的介质中。 计算机用…

Jenkins基础篇--凭据(Credential)管理

什么是凭据 Jenkins的Credentials直译为证书、文凭&#xff0c;我们可以理解为它是钥匙&#xff0c;用来做某些事情的认证。 如Jenkins 和 GitLab交互时&#xff0c;需要添加GitLab的API令牌和登录凭证。 如Jenkins 添加从节点时&#xff0c;需要添加从节点的登录凭证或者Je…

Linux的DHCP工作原理和dns服务器

目录 一、DHCP原理 1.DHCP的好处 2.DHCP的分配方式 3.实验 二、dns服务器 1.什么是dns 2.dns域名解析 3、在内网搭建dns 一、DHCP原理 DCHP工作原理使用C/S架构 &#xff08;1&#xff09;第一步&#xff0c;客户端广播发送一个discover报文寻找DHCP服务器。 &#…

网络爬虫丨基于requests+mysql爬取猫眼热门电影数据做可视化分析

文章目录 写在前面实验描述实验内容遇到问题 写在后面 写在前面 本期内容&#xff1a;基于requestsmysql爬取猫眼热门电影数据做可视化分析 实验需求 anaconda丨pycharmpython3.11.4requestsmysql 项目下载地址&#xff1a;https://download.csdn.net/download/m0_68111267…

OceanBase 4.2特性解读:Show Trace全链路跟踪,助力快速问题定位与精准诊断

在分布式数据库环境下&#xff0c;慢 SQL 诊断是运维人员面临的一大挑战。在无法及时发现问题根本原因的情况下&#xff0c;可能会严重影响用户体验&#xff0c;甚至会导致业务服务不可用。相对于单机数据库&#xff0c;分布式数据库系统涉及多个节点、多组件的协同工作&#x…

苍穹外卖学习----出错记录

1.微信开发者工具遇到的问题&#xff1a; 1.1appid消失报错&#xff1a; {errMsg: login:fail 系统错误,错误码:41002,appid missing [20240112 16:44:02][undefined]} 1.2解决方式&#xff1a; appid可在微信开发者官网 登录账号后在开发栏 找到 复制后按以下步骤粘贴即…