《Vite 基础知识》关于 .mjs .cjs 文件引出 NodeJS 对JS模块加载的思考(CommonJS 和 ESM)

本文主要是介绍《Vite 基础知识》关于 .mjs .cjs 文件引出 NodeJS 对JS模块加载的思考(CommonJS 和 ESM),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

学习 Webpack/Vue2 升级 Vite/Vue3 时,发现以下不同:

  1. 新建的 Vitepress 项目默认创建了 config.mjs 文件;

  2. 新建的 Vite/Vue3 项目,package.json 中默认加上 type: 'module' 配置;

  3. 新建的 Vite/Vue3 项目,postcss.config.cjs 文件后缀必须是 .cjs,否则报错;

其实它们是一个问题,Node.js .js 文件的模块加载方式!

分析

先理清几个概念!!!

Node.js 的模块加载方法有两种:CommonJS ES Modules(ES6/ESM,接下来都用简称 ESM )

CommonJS 和 ESM 区别

功能CommonJSESM
导入/导出接口require()exports/module.exportsimportexport
输出值的拷贝值的引用
动态导入不支持支持
作用域全局作用域局部作用域
加载运行时加载编译时输出接口
同步/异步require() 是同步加载模块import 命令是异步加载模块
循环依赖程序复杂时容易崩溃可处理循环依赖,因为是静态作用域
浏览器兼容支持现代浏览器,旧版不支持更适用于早期 Node.js 环境
Node.js 支持Node.js打包 JS 代码的原始方式v8.5.0(2017.9.12) 开始支持
v13.2.0(2019.11.21) 开始默认支持

值得学习的参考链接

Node.js 指导(暂无中文官网,网上找到的其它中文版本较旧,建议直接看英文最新版):

  • NodeJS CommonJS 和 ESM

  • NodeJS CommonJS 中的 require 和 exports/module.exports 说明;

  • NodeJS ESM 中的 import and export 使用官方语法,推荐中文import 和 export;

  • 关于CommonJSESM 区别说明,推荐中文 《ES6 模块与 CommonJS 模块的差异》;

  • 关于CommonJSESM 互操作性说明;

  • 循环加载 《CommonJS 模块的循环加载》 和 《ES6 模块的循环加载》;

  • Javascript 静态和动态作用域;

package.json 属性 type

package.json 是对 Node.js 项目或 npm 包的描述,里面包含许多元信息。如:项目名称,版本,入口文件,贡献者、依赖插件等。

type 属性值:

  • commonjs(默认):项目中的 .js 文件都作为 CommonJS 模块加载;
  • module:项目中的 .js 文件都作为 ESM 模块加载;
{// 默认加载方式,不设置也是 commonjstype: 'commonjs',// ESM 方式加载type: 'module',
}

也可混合使用,无论 package.json 中设置哪种 type

// 后缀为 .cjs 都将以 CommonJS 方式加载
import './legacy-file.cjs';// 后缀为 .cjs 都将以 ESM 方式加载
import 'commonjs-package/src/index.mjs';

值得学习的参考链接

  • CommonJS 模块加载 ES6 模块
  • ES6 模块加载 CommonJS 模块
  • 同时支持两种格式的模块

浏览器加载 ESM 模块

要加入type="module"属性,且默认是 defer,即 DOM 渲染完再执行。

<script type="module" src="./foo.js"></script>
<!-- 等同于 -->
<script type="module" src="./foo.js" defer></script>

最后

解答 前言 提到的3个不同点!

  1. Vitepress 文件 config.mjs

创建 Vitepress 项目时,package.json 文件中默认没有配置属性 typeVite 相关项目默认支持 ESM 加载方式,所以创建了文件 .mjs ,以支持 ESM 加载;

  1. Vite/Vue3 项目 postcss.config.cjs

postcss 使用后缀 .js 报如下错。因为 Vite 官网 明确指出 postcss 配置文件暂不支持 ES6 module,必须明确使用后缀 .cjs 也就是 CommonJS 方式加载。

ReferenceError: module is not defined in ES module scope
  1. Vite/Vue3 项目 package.json 属性 type: 'module'

Vite 以 原生 ESM 方式提供源码,加载 .js文件默认使用 ESM 方式。且注意 Vite 的 CJS Node API 构建已经被废弃,并将在 Vite 6 中移除。

PS

整理上述内容时,不同文档有如下三种叫法,其实是一个东西,请注意!

  • ESM
  • ES Modules
  • ES6 Module

这篇关于《Vite 基础知识》关于 .mjs .cjs 文件引出 NodeJS 对JS模块加载的思考(CommonJS 和 ESM)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/757983

相关文章

python logging模块详解及其日志定时清理方式

《pythonlogging模块详解及其日志定时清理方式》:本文主要介绍pythonlogging模块详解及其日志定时清理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录python logging模块及日志定时清理1.创建logger对象2.logging.basicCo

JS+HTML实现在线图片水印添加工具

《JS+HTML实现在线图片水印添加工具》在社交媒体和内容创作日益频繁的今天,如何保护原创内容、展示品牌身份成了一个不得不面对的问题,本文将实现一个完全基于HTML+CSS构建的现代化图片水印在线工具... 目录概述功能亮点使用方法技术解析延伸思考运行效果项目源码下载总结概述在社交媒体和内容创作日益频繁的

Node.js 数据库 CRUD 项目示例详解(完美解决方案)

《Node.js数据库CRUD项目示例详解(完美解决方案)》:本文主要介绍Node.js数据库CRUD项目示例详解(完美解决方案),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考... 目录项目结构1. 初始化项目2. 配置数据库连接 (config/db.js)3. 创建模型 (models/

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

Qt spdlog日志模块的使用详解

《Qtspdlog日志模块的使用详解》在Qt应用程序开发中,良好的日志系统至关重要,本文将介绍如何使用spdlog1.5.0创建满足以下要求的日志系统,感兴趣的朋友一起看看吧... 目录版本摘要例子logmanager.cpp文件main.cpp文件版本spdlog版本:1.5.0采用1.5.0版本主要

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

Spring Boot 配置文件之类型、加载顺序与最佳实践记录

《SpringBoot配置文件之类型、加载顺序与最佳实践记录》SpringBoot的配置文件是灵活且强大的工具,通过合理的配置管理,可以让应用开发和部署更加高效,无论是简单的属性配置,还是复杂... 目录Spring Boot 配置文件详解一、Spring Boot 配置文件类型1.1 applicatio

Python使用date模块进行日期处理的终极指南

《Python使用date模块进行日期处理的终极指南》在处理与时间相关的数据时,Python的date模块是开发者最趁手的工具之一,本文将用通俗的语言,结合真实案例,带您掌握date模块的六大核心功能... 目录引言一、date模块的核心功能1.1 日期表示1.2 日期计算1.3 日期比较二、六大常用方法详

SpringBoot项目启动报错"找不到或无法加载主类"的解决方法

《SpringBoot项目启动报错找不到或无法加载主类的解决方法》在使用IntelliJIDEA开发基于SpringBoot框架的Java程序时,可能会出现找不到或无法加载主类com.example.... 目录一、问题描述二、排查过程三、解决方案一、问题描述在使用 IntelliJ IDEA 开发基于

python中time模块的常用方法及应用详解

《python中time模块的常用方法及应用详解》在Python开发中,时间处理是绕不开的刚需场景,从性能计时到定时任务,从日志记录到数据同步,时间模块始终是开发者最得力的工具之一,本文将通过真实案例... 目录一、时间基石:time.time()典型场景:程序性能分析进阶技巧:结合上下文管理器实现自动计时