使用 esbuild 混合插件为多个文件打包 IIFE 和单个 ESM 包

使用 esbuild 混合插件为多个文件打包 iife 和单个 esm 包

本文介绍如何使用 esbuild 插件和 `define` 特性,为 JavaScript 项目同时生成 IIFE (Immediately Invoked Function Expression) 和 ESM (ECMAScript Module) 两种格式的包。通过自定义插件移除 IIFE 构建中的 imports,并利用 `define` 标志在代码中区分不同构建环境,最终实现代码的按需引入和精简输出,从而优化构建产物的大小和性能。

背景

在开发 JavaScript 库或框架时,通常需要支持多种模块化格式,以满足不同用户的需求。例如,一些用户可能仍然喜欢使用传统的 标签引入 IIFE 格式的包,而另一些用户则更倾向于使用 ESM 格式的包,以便利用现代 JavaScript 的特性,如 tree shaking。

本文将以 SlickGrid 项目为例,介绍如何使用 esbuild 插件和 define 特性,同时生成 IIFE 和 ESM 两种格式的包,并确保每种格式的包都只包含必要的代码,从而优化构建产物的大小和性能。

解决方案

该解决方案的核心在于使用 esbuild 插件来处理 IIFE 构建中的 imports,并使用 define 特性在代码中区分不同的构建环境。

1. 自定义 esbuild 插件移除 Imports

首先,我们需要创建一个 esbuild 插件,用于在 IIFE 构建中移除所有的 import 语句。这是因为 IIFE 格式的包通常依赖于全局变量,而不是模块导入。

import { build } from 'esbuild';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 和 onLoad 钩子。onResolve 钩子拦截所有非入口文件的导入请求,并将其重定向到 import-ns 命名空间。onLoad 钩子则拦截 import-ns 命名空间下的所有文件加载请求,并返回一个空字符串,从而有效地移除了所有的 import 语句。

2. 使用 define 特性区分构建环境

接下来,我们需要使用 esbuild 的 define 特性,在代码中定义一个全局变量,用于区分 IIFE 和 ESM 构建环境。

豆包爱学 豆包爱学

豆包旗下AI学习应用

豆包爱学 674 查看详情 豆包爱学

/** 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.jsexport function buildEsm() {  build({    entryPoints: ['index.js'],    format: 'esm',    target: 'es2020',    treeShaking: true,    define: { IIFE_ONLY: 'false' },    outdir: `dist/esm`,  });}

在 IIFE 构建中,我们将 IIFE_ONLY 定义为 ‘true’,而在 ESM 构建中,我们将 IIFE_ONLY 定义为 ‘false’。

3. 在代码中使用 IIFE_ONLY 变量

现在,我们可以在代码中使用 IIFE_ONLY 变量来区分不同的构建环境。例如:

// imports will be auto-dropped in iife by custom pluginimport { 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);

在 IIFE 构建中,IIFE_ONLY 为 ‘true’,因此 SlickEvent 和 Utils 将从全局 Slick 对象中获取。而在 ESM 构建中,IIFE_ONLY 为 ‘false’,因此 SlickEvent 和 Utils 将从 import 语句中获取。

示例

以下是一个完整的示例,展示如何使用 esbuild 插件和 define 特性同时生成 IIFE 和 ESM 格式的包:

// esbuild.config.jsimport { build } from 'esbuild';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',      }));    }};/** 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.jsexport function buildEsm() {  build({    entryPoints: ['index.js'],    format: 'esm',    target: 'es2020',    treeShaking: true,    define: { IIFE_ONLY: 'false' },    outdir: `dist/esm`,  });}// index.js// imports will be auto-dropped in iife by custom pluginimport { 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);

总结

通过使用 esbuild 插件和 define 特性,我们可以轻松地为 JavaScript 项目同时生成 IIFE 和 ESM 两种格式的包,并确保每种格式的包都只包含必要的代码。这可以显著减小构建产物的大小,并提高应用程序的性能。

注意事项:

define 特性会将字符串值视为 JavaScript 代码,因此需要确保字符串值的格式正确。在 IIFE 构建中,需要确保所有的依赖项都已通过 标签加载到全局环境中。可以根据实际需求调整插件和 define 特性的配置。

希望本文能够帮助你更好地理解如何使用 esbuild 同时生成 IIFE 和 ESM 格式的包。

以上就是使用 esbuild 混合插件为多个文件打包 IIFE 和单个 ESM 包的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月5日 05:32:15
下一篇 2025年11月5日 05:35:37

相关推荐

  • Symfony 如何把图片资源转为数组

    获取图片元数据:使用 exif_read_data() 或 getimagesize() 函数提取图片的宽度、高度、mime 类型等信息并存入数组;2. 将图片编码为 base64:通过 file_get_contents() 读取图片内容并用 base64_encode() 转换为字符串,存入数组…

    2025年12月11日
    000
  • VSCode怎样设置PHP文件的默认打开方式 VSCode新手设置PHP文件默认打开的简单步骤​

    将vscode设为php文件默认打开方式最直接的方法是通过操作系统设置:windows系统右键php文件选择“打开方式”并勾选“始终使用此程序”;macos在“显示简介”中更改“打开方式”并点击“全部更改”;linux在文件属性中设置默认应用。2. vscode内部的files.associatio…

    2025年12月11日
    000
  • 如何使用Docker调试PHP后端接口 PHP本地接口测试配置方法

    要实现在docker中调试php后端接口,核心在于正确配置xdebug与ide通信,具体步骤如下:1. 使用docker-compose.yml定义php-fpm、nginx及数据库服务,并设置代码挂载与端口映射;2. 在dockerfile中安装xdebug并配置php.ini,确保client_…

    2025年12月11日 好文分享
    000
  • 如何用Docker搭建多版本PHP环境 PHP项目版本隔离部署方法

    通过docker搭建多版本php环境,实现项目版本隔离部署,核心是利用容器的隔离性为每个项目创建独立运行环境。1. 选择合适的基础镜像(如php:7.4-fpm-alpine或php:8.1-fpm-alpine);2. 在dockerfile中安装项目所需扩展(如pdo_mysql、opcache…

    2025年12月11日 好文分享
    000
  • 利用PHPMyAdmin执行SQL语句创建数据库视图

    登录phpmyadmin并选择目标数据库;2. 点击顶部“sql”标签进入执行界面;3. 编写create view语句,例如:create view view_name as select columns from table where condition;4. 输入具体视图定义,如包含单表筛选…

    2025年12月11日 好文分享
    000
  • 配置PhpStorm代码折叠和展开的规则

    phpstorm 的代码折叠功能可通过设置和快捷键开启或关闭,并支持按语言结构自定义折叠规则,同时提供快捷键与鼠标操作实现高效代码浏览。具体包括:1. 在 settings 中勾选 enable code folding 或使用快捷键切换状态;2. 在 code folding 设置项中启用或禁用不…

    2025年12月11日 好文分享
    000
  • PHP命名空间:组织代码结构

    php命名空间用于解决类名、函数名等标识符冲突问题,并提升代码可读性与维护性。1.命名空间通过逻辑分组避免冲突,如同不同文件夹允许同名文件;2.使用namespace声明命名空间,如namespace myappmodels;3.引用类时可用fqn或use关键字导入简化;4.支持子命名空间嵌套,如m…

    2025年12月11日 好文分享
    000
  • 配置PhpStorm代码格式化的规则和快捷键

    配置phpstorm的代码格式化规则和快捷键需先选择语言规范并设置代码风格,再自定义细节规则,最后配置快捷键及自动保存选项。首先打开settings进入editor > code style选择对应语言并新建或复制配置方案,可导入.editorconfig或psr-12标准,也可手动调整缩进、…

    2025年12月11日 好文分享
    000
  • 使用phpMyAdmin快速创建和管理数据库表

    phpmyadmin是一个基于web的mysql数据库管理工具,它提供图形界面,简化数据库操作。使用它创建数据库表的方法是:1. 选择数据库;2. 点击“新建”按钮;3. 定义表名、字段名、数据类型和长度等;4. 点击“保存”。phpmyadmin将操作转换成sql语句执行,同时支持数据导入导出和表…

    2025年12月11日
    000
  • PHP函数代码风格的在线资源

    PHP 函数代码风格的在线资源 保持一致的代码风格对于代码可读性和可维护性至关重要。对于 PHP,有一些在线资源可以帮助您遵守最佳实践。 PHP_CodeSniffer PHP_CodeSniffer 是一款静态分析工具,可根据一组预定义的规则检查 PHP 代码。它可以检测编码标准违规并建议修复。您…

    2025年12月10日
    000
  • php函数跨语言调用实战指导

    #%#$#%@%@%$#%$#%#%#$%@_e1bfd762321e409c++ee4ac0b6e841963c 可通过外部函数接口(ffi)实现与其他语言的跨语言调用。实战案例:安装 ffi 扩展定义 c++ 函数签名加载 c++ 函数库使用 ffi 库调用 c++ 函数,实现从 php 调用其…

    2025年12月10日
    000
  • 使用linter工具实现PHP函数参数类型检查

    通过使用linter工具phpstan,我们可以实现php函数参数的类型检查。phpstan是一种静态分析工具,可通过分析变量类型的推断来检查函数参数类型。我们可以使用composer安装phpstan并通过配置phpstan.neon文件来设置检查级别。phpstan通过类型断言和严格类型检查来检…

    2025年12月10日
    000
  • 稳定币交易所最新排行榜top10

    数字货币交易领域日新月异,稳定币作为连接法币与加密货币的桥梁,其重要性日益凸显。众多交易平台在提供稳定币交易服务的同时,也在不断优化用户体验和安全性。本文将为您梳理当前市场上表现突出的十大稳定币交易所,并对其进行简要介绍,旨在为您的交易决策提供参考。 1. 欧易OKX 作为全球领先的数字资产交易平台…

    2025年12月10日 好文分享
    000
  • 稳定币十大交易平台最新排名(2025最新排行榜)

    在数字资产飞速发展的2025年,稳定币作为连接法币与加密货币世界的桥梁,其重要性日益凸显。稳定币交易平台的排名直接关系到用户的资产安全、交易效率以及可获得的金融服务。本篇文章将基于最新的市场动态和用户反馈,为您呈现2025年稳定币交易平台的十大最新排名,帮助您在纷繁复杂的市场中做出明智的选择。 稳定…

    2025年12月10日 好文分享
    000
  • 数字币交易平台app十大排行榜

    数字货币交易平台作为加密资产流通和交易的核心场所,在全球范围内吸引了大量用户。随着数字货币市场的不断发展,交易平台的数量和规模也日益增长,为用户提供了多样化的选择。本文旨在梳理当前市场上备受关注的数字币交易平台,并根据其用户基础、交易量、安全性、创新性等多个维度进行综合评估,为您呈现一份客观的十大排…

    2025年12月10日 好文分享
    000
  • 十大虚拟货币交易所排行榜2025最新

    加密货币的世界日新月异,交易平台的竞争也日益激烈。对于广大投资者而言,选择一个安全、稳定、功能齐全的交易平台至关重要。本文将基于2025年的最新数据,为您呈现十大虚拟货币交易所的排行榜,帮助您更好地 navigating the digital asset market. 以下是2025年最新的十大…

    2025年12月10日 好文分享
    000
  • Pendle($PENDLE)是什么?PENDLE价格分析及2025年预测

    目录 Pendle的概念和特点1.基本结构2. 使用示例3. 多链支持和流动性基础设施4. 治理与代币经济5. 区别点PENDLE 的当前价格和近期趋势2025年市场预测1.保守情景(维持5-6美元)2. 中性情景(上涨 6-8 美元)3. 激进情景(突破 10 美元)PENDLE生长因子1. 扩大…

    2025年12月10日
    000
  • BNB币2025年价格预测:突破历史新高后,BNB会达到$2,000吗?

    目录 BNB Chain 及其原生代币BNB 的代币经济学是什么?BNB 的主要用途包括:是什么推动BNB 在2025 年10 月创下历史新高?1. 链上活动激增,推动BNB 代币销毁创纪录2. 机构采用将BNB 确立为企业储备资产3. BNB Chain 生态系统增长巩固长期基础4. 市场动能和恢…

    2025年12月10日 好文分享
    000
  • 加密货币投资必读

    在数字时代浪潮中,加密货币以其独特的魅力吸引着全球投资者的目光。它不仅仅是一种新兴的金融资产,更代表着底层技术革新——区块链的巨大潜力。然而,如同任何一种投资形式,加密货币市场也充满了机遇与风险。对于希望踏入这片未知领域的投资者而言,掌握必要的知识与策略,规避潜在陷阱,显得尤为重要。本文将深入探讨加…

    好文分享 2025年12月10日
    000
  • Bitfinex:专业交易

    在加密货币交易的浩瀚宇宙中,bitfinex无疑是其中一颗耀眼的星辰。它不仅仅是一个简单的交易所,更是一个为专业交易者量身定制的复杂生态系统。踏入bitfinex的大门,你将发现一个集高流动性、先进交易工具、深度市场数据以及强大安全保障于一体的交易殿堂。这里汇聚了全球顶级的机构投资者、资深交易员以及…

    好文分享 2025年12月10日
    000

发表回复

登录后才能评论
关注微信