目录

  • 概要
  • 文件优先级
  • browser、module 和 main
    • 字段定义
    • 使用场景与优先级
      • webpack + web + ESM
      • webpack + web + commonJS
      • webpack + node + ESM/commonJS
      • node + commonJS
      • node + ESM
  • 总结

概要

前端开发中使用到 npm 包那可算是家常便饭,而使用到 npm 包总免不了接触到 package.json 包配置文件。

那么这里就有一个问题,当我们在不同环境下 import 一个 npm 包时,到底加载的是 npm 包的哪个文件?

老司机们很快地给出答案:main 字段中指定的文件

然而我们清楚 npm 包其实又分为:

  • 只允许在客户端使用的,
  • 只允许造服务端使用的,
  • 浏览器/服务端都可以使用。

如果我们需要开发一个 npm 包同时兼容支持 web端 和 server 端,需要在不同环境下加载npm包不同的入口文件,显然一个 main 字段已经不能够满足我们的需求,这就衍生出来了 modulebrowser 字段。

本文就来说下 这几个字段的使用场景,以及同时存在这几个字段时,他们之间的优先级。

文件优先级

在说 package.json 之前,先说下文件优先级

由于我们使用的模块规范有 ESM 和 commonJS 两种,为了能在 node 环境下原生执行 ESM 规范的脚本文件,.mjs 文件就应运而生。

当存在 index.mjsindex.js 这种同名不同后缀的文件时,import './index' 或者 require('./index') 是会优先加载 index.mjs 文件的。

也就是说,优先级 mjs > js

browser、module 和 main

字段定义

  • main : 定义了 npm 包的入口文件,browser 环境和 node 环境均可使用
  • module : 定义 npm 包的 ESM 规范的入口文件,browser 环境和 node 环境均可使用
  • browser : 定义 npm 包在 browser 环境下的入口文件

使用场景与优先级

首先,我们假定 npmtest 有以下目录结构

──── lib|── index.browser.js|── index.browser.mjs|── index.js|── index.mjs

其中 *.js 文件是使用 commonJS 规范的语法(require('xxx')),*.mjs 是用 ESM 规范的语法(import 'xxx')

其 package.json 文件:

  "main": "lib/index.js",  // main "module": "lib/index.mjs", // module// browser 可定义成和 main/module 字段一一对应的映射对象,也可以直接定义为字符串// "browser": "./lib/index.browser.js", // browser"browser": {"./lib/index.js": "./lib/index.browser.js", // browser+cjs"./lib/index.mjs": "./lib/index.browser.mjs"  // browser+mjs}

根据上述配置,那么其实我们的 package.json 指定的入口可以有

  • main
  • module
  • browser
  • browser+cjs
  • browser+mjs

这 5 种情况。下面说下具体使用场景。

webpack + web + ESM

这是我们最常见的使用场景,通过 webpack 打包构建我们的 web 应用,模块语法使用 ESM

当我们加载

import test from 'test'

实际上的加载优先级是 browser = browser+mjs > module > browser+cjs > main
也就是说 webpack 会根据这个顺序去寻找字段指定的文件,直到找到为止。

然而实际上的情况可能比这个更加复杂,具体可以参考流程图

webpack + web + commonJS

const test = require('test')

事实上,构建 web 应用时,使用 ESM 或者 commonJS 模块规范对于加载优先级并没有任何影响

优先级依然是 browser = browser+mjs > module > browser+cjs > main

webpack + node + ESM/commonJS

我们清楚,使用 webpack 构建项目的时候,有一个 target 选项,默认为 web,即进行 web 应用构建。

当我们需要进行一些 同构项目,或者其他 node 项目的构建的时候,我们需要将 webpack.config.jstarget 选项设置为 node 进行构建。

import test from 'test'
// 或者 const test = require('test')

优先级是: module > main

node + commonJS

通过 node test.js 直接执行脚本

const test = require('test')

只有 main 字段有效。

node + ESM

通过 --experimental-modules 可以让 node 执行 ESM 规范的脚本(必须是 mjs 文件后缀)
`node --experimental-modules test.mjs

import test from 'test'

只有 main 字段有效。

总结

  • 如果 npm 包导出的是 ESM 规范的包,使用 module
  • 如果 npm 包只在 web 端使用,并且严禁在 server 端使用,使用 browser。
  • 如果 npm 包只在 server 端使用,使用 main
  • 如果 npm 包在 web 端和 server 端都允许使用,使用 browser 和 main
  • 其他更加复杂的情况,如npm 包需要提供 commonJS 与 ESM 等多个规范的多个代码文件,请参考上述使用场景或流程图

作者:七月流萤
链接:https://juejin.cn/post/6844903862977953806
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

package.json 中的 browser、module、main 字段使用探索相关推荐

  1. package.json 中的 Module 字段是干嘛的

    引入 最近团队的一个同学在搞 npm library 源码的调试插件,因为内部的一个组件库含有大量的逻辑,在某个项目中不经意就出现一个磨人的 bug,但是组件库发布都是打包编译后的代码,而且没有 pu ...

  2. package.json中bin字段的用处

    如下是一段package.json中的代码,其中可以看到版本号等信息,除此之外还能看到一个bin字段,本文主要讲的就是bin字段的用处 {"name": "mys-gj& ...

  3. package.json 中的 bin 字段作用

    我们在下载 npm 包后,查看 npm 包的 package.json 有些包会有 bin 字段 例如我们常用的 webpack-cli 包 那么 bin 字段有什么用呢? 官方文档链接 许多软件包都 ...

  4. 【前端工程化】配置package.json中scripts命令脚本,新手必学

    每日鸡汤:你总要努力追上那个曾经被赋予众望的自己吧 目录 前言 一.运行npm run 命令之后会干啥? 1. scripts里面写啥 2. node_modules/.bin 二进制可执行文件 二. ...

  5. 在package.json中使用git URL依赖分支或标记?

    本文翻译自:Depend on a branch or tag using a git URL in a package.json? Say I've forked a node module wit ...

  6. npm更新模块并同步到package.json中

    使用原始npm 1.查看需要更新的版本 npm outdated 该命令会列出所有需要更新的项目 2.修改package.json中需要更新的包对应的版本号 npm update 由于npm upda ...

  7. Node.js中package.json中库的版本号详解(^和~区别)

    Node.js中package.json中库的版本号详解(^和~区别) 当我们查看package.json中已安装的库的时候,会发现他们的版本号之前都会加一个符号,有的是插入符号(^),有的是波浪符号 ...

  8. android中json插件,【Android原生插件】package.json中关于第三方aar的配置

    按照文档(https://ask.dcloud.net.cn/article/35414)所说: dependencies节点特殊说明 android插件中集成的第三方SDK 如果是jar或so放入到 ...

  9. package.json 中的波浪号(~)和插入符号(^)有什么区别?

    问题描述: 在我升级到最新的稳定版 node 和 npm 后,我尝试了 npm install moment --save.它使用插入符号 ^ 前缀将条目保存在 package.json 中.以前,它 ...

最新文章

  1. C语言实例第3期:在控制台打印出著名的杨辉三角
  2. Full_of_Boys训练6总结
  3. file 选择的图片作为背景图片_酷炫!用Python把桌面变成实时更新的地球图片
  4. [J2ME]RSSOwlMidlet(RSS无线阅读器)设计说明
  5. 树莓派4B Raspbian-buster 更换源
  6. 挑战10个最难的Java面试题(附答案)【上】
  7. 为什么遇见逆水寒服务器维修,《遇见逆水寒》4月23日更新公告
  8. Address already in use
  9. 提高局域网速度的21招
  10. 2015最新Linkedin人才趋势报告
  11. ora-01045 :user system lacks create session privilege; logon denied
  12. 分享一个无意间发现的躺赚网络创业小项目!
  13. 阿里云服务器ddos攻击防御
  14. 怎样自己创建一个个人网站,怎样将自己写的网站发布到外网?
  15. win10企业版2016长期服务激活教程
  16. 硅谷丛林的故事 EDA篇
  17. HIFIVE音乐开放平台音乐api接口文档!
  18. Blender游戏开发教程
  19. 一文简单了解互联网流量变现
  20. 计算语言学(CL)与自然语言处理(NLP)

热门文章

  1. day7-字典作业(1)
  2. 成都榆熙教育:拼多多新手开店商品类目怎么选择?
  3. 设置两个无线路由器同时无线上网的方法
  4. 【游戏运营】【实战】属性养成分析——王国纪元
  5. 服务器安全策略经验总结
  6. 异常检测与故障诊断的区别
  7. 电脑底部工具栏跑到侧面了怎么办
  8. iOS 高性能置灰方案
  9. 86 - 得到整数列表的中位数
  10. python字符串输入语句_Python 字符串与基本语句