0

0

使用 Esbuild 插件和 Define 实现 IIFE 和 ESM 混合打包

花韻仙語

花韻仙語

发布时间:2025-10-29 16:53:02

|

571人浏览过

|

来源于php中文网

原创

使用 esbuild 插件和 define 实现 iife 和 esm 混合打包

本文介绍如何使用 Esbuild 插件和 `define` 功能,针对既有 IIFE 模块又有 ESM 模块的项目,实现同时生成多个独立的 IIFE 文件和一个 ESM 打包文件的混合打包方案。通过自定义 Esbuild 插件移除 IIFE 构建中的 import 语句,并利用 `define` 功能在不同构建目标中切换代码逻辑,最终实现更简洁、更高效的打包流程。

背景

在一些老项目中,可能同时存在使用 IIFE (Immediately Invoked Function Expression) 编写的模块和使用 ESM (ECMAScript Modules) 编写的模块。IIFE 模块通常依赖全局变量,而 ESM 模块则使用 import 和 export 进行模块化。如何使用 Esbuild 同时构建这两种类型的模块,是一个值得探讨的问题。

解决方案

核心思路是利用 Esbuild 的插件机制和 define 功能,针对不同的构建目标(IIFE 和 ESM)进行定制化处理。

  1. 移除 IIFE 构建中的 import 语句: 通过自定义 Esbuild 插件,拦截 import 语句,并将其替换为空字符串。由于 IIFE 模块依赖全局变量,因此不需要 import 语句。

  2. 使用 define 功能切换代码逻辑: 在代码中使用 define 定义的变量,例如 IIFE_ONLY,根据构建目标的不同,赋予不同的值。在 IIFE 构建中,IIFE_ONLY 为 true,在 ESM 构建中,IIFE_ONLY 为 false。这样,就可以根据不同的构建目标,执行不同的代码逻辑。

具体实现

1. 自定义 Esbuild 插件

以下代码展示了如何创建一个 Esbuild 插件,用于移除 IIFE 构建中的 import 语句:

const removeImportsPlugin = {
  name: 'remove-imports-plugin',
  setup(build) {
    build.onResolve({ filter: /.*/ }, (args) => {
      if (args.kind !== 'entry-point') {
        return { path: args.path + '.js', namespace: 'import-ns' }
      }
    });
    build.onLoad({ filter: /.*/, namespace: 'import-ns' }, () => ({
      contents: `// empty string, do nothing`,
      loader: 'js',
    }));
  }
};

这个插件包含两个部分:

  • onResolve: 拦截所有非入口文件的引用,并将其指向一个带有 import-ns 命名空间的虚拟文件。
  • onLoad: 加载 import-ns 命名空间的文件,并将其内容替换为空字符串。

2. 使用 define 功能

在代码中使用 define 定义的变量,例如 IIFE_ONLY,根据构建目标的不同,赋予不同的值。例如:

Closers Copy
Closers Copy

营销专用文案机器人

下载
// imports will be auto-dropped in iife by custom plugin
import { SlickEvent as SlickEvent_, Utils as Utils_ } from '../slick.core';

// for (iife) load `Slick` methods from global window object, or use imports for (cjs/esm)
const SlickEvent = IIFE_ONLY ? Slick.Event : SlickEvent_;
const Utils = IIFE_ONLY ? Slick.Utils : Utils_;

// ...

// then use it normally in the code...
const options = Utils.extend(true, {}, defaults, options);

3. 构建脚本

以下代码展示了如何使用 Esbuild 构建 IIFE 和 ESM 模块:

import { build } from 'esbuild';

/** build as iife, every file will be bundled separately */
export async function buildIifeFile(file) {
  build({
    entryPoints: [file],
    format: 'iife',
    // add Slick to global only when filename `slick.core.js` is detected
    globalName: /slick.core.js/.test(file) ? 'Slick' : undefined,
    define: { IIFE_ONLY: 'true' },
    outfile: `dist/browser/${file.replace(/.[j|t]s/, '')}.js`,
    plugins: [removeImportsPlugin],
  });
}

// bundle in ESM format into single file index.js
export function buildEsm() {
  build({
    entryPoints: ['index.js'],
    format: 'esm',
    target: 'es2020',
    treeShaking: true,
    define: { IIFE_ONLY: 'false' },
    outdir: `dist/esm`,
  });
}

在构建 IIFE 模块时,需要指定 format: 'iife',并设置 define: { IIFE_ONLY: 'true' },同时使用 removeImportsPlugin 移除 import 语句。 globalName 选项可以设置全局变量名,只有核心文件需要设置。

在构建 ESM 模块时,需要指定 format: 'esm',并设置 define: { IIFE_ONLY: 'false' }。

示例

假设有以下代码:

// plugins/slick.cellcopymanager.js
import { SlickEvent as SlickEvent_, Utils as Utils_ } from '../slick.core';

// for (iife) load `Slick` methods from global window object, or use imports for (cjs/esm)
const SlickEvent = IIFE_ONLY ? Slick.Event : SlickEvent_;
const Utils = IIFE_ONLY ? Slick.Utils : Utils_;

function CellCopyManager() {
  // ...
}

构建后的 IIFE 模块代码如下:

"use strict";
(() => {
  // plugins/slick.cellcopymanager.js
  var SlickEvent = Slick.Event, Utils = Slick.Utils;
  function CellCopyManager() {
// ...

构建后的 ESM 模块代码如下:

// plugins/slick.cellcopymanager.js
var SlickEvent5 = SlickEvent, Utils10 = Utils;
function CellCopyManager() {
// ...

总结

通过使用 Esbuild 插件和 define 功能,可以轻松地实现 IIFE 和 ESM 混合打包。这种方案不仅能够生成符合不同规范的模块,还能够避免冗余代码,提高构建效率。

注意事项

  • 在代码中使用 define 定义的变量时,需要确保变量名是唯一的,避免与其他变量冲突。
  • 在构建脚本中,需要根据实际情况调整 Esbuild 的配置选项。
  • 该方案适用于需要同时支持 IIFE 和 ESM 两种模块规范的项目。

相关专题

更多
typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

103

2023.09.26

define的用法
define的用法

define用法:1、定义常量;2、定义函数宏:3、定义条件编译;4、定义多行宏。更多关于define的用法的内容,大家可以阅读本专题下的文章。

315

2023.10.11

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

617

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

427

2024.06.27

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

73

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

96

2025.09.18

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

249

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.04

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号