Webpack 什么是loader?什么是plugin?loader与plugin区别是什么?

什么是loader?什么是plugin?

  • loader 本质为一个函数,将文件编译成可执行文件。webpack完成的工作是将依赖分析与tree shinking对于类似.vue或.scss结尾的文件无法编译理解这就需要实现一个loader完成文件转译成js、html、css、json等可执行文件。
  • plugin可以是函数或Class(es6),通过订阅webpack串行流程上的事件,实现扩展webpack功能,改变webpack输出结果。

区别:

  1. 执行时机不同。loader主要是倒序执行,从后往前将一个loader的输出作为它前一个loader的输入,直到运行完所有loader;plugin执行执行时机在与通过订阅webpack串行流程上的事件,当webpack运行到对应订阅事件时执行,可以运行在整个生命周期内。两者都可以同步或者异步执行。
  2. 侧重功能不同。loader主要完成文件操作;plugin主要是实现除loader以外功能,扩展webpack功能。
  3. 配置方式不同。loader配置到module.rules,rules是一个数组,每一个元素是一个对象,每个对象包含属性test(检验目标文件类型)、use:(loader数组,从后往前执行)、options(配置);plugin放在plugins数组中通过new 关键字创建。
  4. 书写方式不同。

​ loader编写伪代码如下

js 复制代码
/**
 * 定义一个loader
 * 1.不可为一个箭头函数,Webpack运行时需要指定运行this为Webpack,所以可以调用内置api
 * 2.可以异步返回使用this.callback
 * 3.配置的options可以使用this.query获取
 * @param source 模块文件源数据内容
 * @return {*}
 */
module.exports = function (source) {

  // 如果 loader 配置了 options 对象,那么this.query将指向 options
  const options = this.query;

  const doTask = async (source) => {
    //伪代码进行数据转换
    const content = await source2Something(source,options)
    /*
    * this.callback 参数:
    * error:Error | null,当 loader 出错时向外抛出一个 error
    * content:String | Buffer,经过 loader 编译后需要导出的内容
    * sourceMap:为方便调试生成的编译后内容的 source map
    * ast:本次编译生成的 AST 静态语法树,之后执行的 loader 可以直接使用这个 AST,进而省去重复生成 AST 的过程
    */
    this.callback(null, content)
  }

  //异步返回结果
  doTask()
  //同步返回
  return source2Something(source,options)
}

​ plugin编写伪代码如下

js 复制代码
/**
 * 定义一个plugin
 * 1.需要通过new创建可以使用class创建
 * 2.内部需要有一个方法apply调用运行插件功能
 * 
 * 具体可以看
 * https://webpack.docschina.org/contribute/writing-a-plugin/#basic-plugin-architecture
 */

class MyPlugin {
  //创建时可以传入参数
  constructor(options) {
    this.options = options;
  }

  /**
   * 通过compiler获取 webpack 内部的钩子,获取 webpack 打包过程中的各个阶段
   * 钩子分为同步和异步的钩子,异步钩子必须执行对应的回调
   * @param compiler
   */
  apply(compiler){
    //同步hook 最后调用tap
    //emit为hook事件名称
    compiler.hooks.emit.tap('MyPlugin',compilation=>{

      //do somethings
      //do somethings
      //do somethings

    })
    //异步hook 最后调用tapAsync、tapPromise
    //done为hook事件名称
    //tapAsync 异步钩子必须执行对应的回调
    //tapPromise 需要返回Promise
    compiler.hooks.done.tap('MyPlugin', (compilation, callback) => {
      console.log("打包已完成");

      //do somethings
      //do somethings
      //do somethings

      callback();
    })
  }

}
相关推荐
圈圈的熊15 分钟前
vue2 和 vue3的区别
前端·javascript·vue.js
Kika写代码18 分钟前
【基于轻量型架构的WEB开发】课程 作业4 AOP
java·前端·架构
前端与小赵3 小时前
关于 AJAX 与 Promise
前端·ajax·okhttp·promise
啊·贤4 小时前
展开运算符 (...):这是 JavaScript 中的一个语法,用于将数组或对象展开到另一个数组或对象中。
前端·javascript·vue.js
硬汉嵌入式4 小时前
《安富莱嵌入式周报》第345期:开源蓝牙游戏手柄,USB3.0 HUB带电压电流测量,LCR电桥前端模拟,开源微型赛车,RF信号扫描仪,开源无线电收发器
前端·游戏·开源
NiNg_1_2344 小时前
前端 Flex 布局语法详解
前端·css·html
吉吉安4 小时前
前端实现选项多选效果(三层结构)
前端·vue.js·html5
青少儿编程课堂7 小时前
2024年3月电子学会Python等级考试试卷(四级)真题,包含答案
前端·javascript·python·电子学会·电子学会2024年3月真题·电子学会真题
始终奔跑在路上7 小时前
软件测试—HTML
前端·功能测试·html·集成测试·软件工程
Moment7 小时前
在 NodeJs 中如何通过子进程与 Golang 进行 IPC 通信 🙄🙄🙄
前端·后端·go