iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > node.js >js中ESM规范的示例分析
  • 356
分享到

js中ESM规范的示例分析

2024-04-02 19:04:59 356人浏览 泡泡鱼
摘要

js中ESM规范的示例分析,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。前端发展到如今,社区生态已经非常丰富。在无数开源大神的努力下,很多前端

js中ESM规范的示例分析,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

js中ESM规范的示例分析

前端发展到如今,社区生态已经非常丰富。

在无数开源大神的努力下,很多前端开发的痛点(比如「静态类型检查」、「浏览器兼容性」)早已有了事实上的标准解决方案(比如TS、babel)。

然而,在这繁荣之下,有一个日常开发不易感知的问题:

  • 模块化规范的混乱

你可曾遇到过莫名其妙的bug,在多方搜资源,反复验证,耗费数个小时终于发现:

原来是某个包导出的是CJS,而项目使用ESM导致。

比如这个例子:记一次打包压缩报错[1]

如果你觉得这是个很容易发现的问题,再考虑结合上node_modules的层层依赖呢?

这个问题,揭开了模块化规范间斗争与博弈的冰山一角。

作为现代前端工程化的基石,模块化规范有太多值得深究的内容。

我会花几篇文章来讲解模块化规范。本文是第一篇,会围绕模块化规范的演进展开。

正文

如果问十年前的前端最头疼的是什么?一定是浏览器兼容性。

随着babel等编译工具出现,兼容性逐渐被工程化方案解决(es6+编译为ES5)。

不仅是「兼容性」问题,DSL(如JSX、Vue的模版语法)、代码压缩、代码静态检查(TS)等日常开发的刚需都能在工程化方案中找到解决办法。

如果将当今繁荣的前端工程化生态比喻为一座大厦,那大厦的地基一定是「模块化规范」。

现代JS代码都是基于「模块化规范」组织起来,让我们从下往上来看看这座大厦:

js中ESM规范的示例分析

规范的实现依赖于宿主环境,比如浏览器环境实现了EcmaScript Module(后文简称ESM)规范。

Node v12之前支持CommonJS(后文简称CJS)规范,12之后同时支持CJS与ESM。

在「宿主环境」之上,是基于模块化规范实现的「工具集」,比如webpack、vite、vscode生态。

再往上,基于「工具集」提供的api,可以实现各种工程化工具。比如:

  • WEBpack loader

  • VScode plugin

  • babel plugin

再往上,就是开发者自己编写的业务代码。

开发者只需要在工具集中配置好工具,就能为业务代码提供服务。比如:

在VScode(工具集)中配置eslint(工具),就能在开发时获得相应提示

在webpack(工具集)中配置babel loader(工具),就能在开发时使用ES6+语法

可见,理想状态下,在开发者视角是不需要关注底层的「模块化规范」实现的。

规范之争

然而,事物是动态发展的,模块化规范也不是一蹴而就的,让我们回到09年。

美国程序员「Ryan Dahl」创造了node.js项目,将JS用于服务端开发。

js中ESM规范的示例分析

node.js使用CJS[2]标准作为模块化规范。

js中ESM规范的示例分析

有了服务端模块规范(CJS),很自然的,JS开发者们想为客户端(主要是浏览器)提供一种模块化规范。

然而CJS是为服务端设计的。

在服务端,io操作通常能迅速完成,所以CJS规范定义的:

  • 模块加载 --> 模块解析 --> 模块执行

这个流程是作为一个整体同步执行的。

然而在浏览器环境,「模块加载」(即数据请求)通常很耗时。有人曾作出一个形象的比喻:

如果一个CPU周期花费1秒完成,那么文件的网络请求需要花费4年。

js中ESM规范的示例分析

显然浏览器端需要一种「支持异步」的模块化规范。

AMD(Asynchronous Module Definition 异步模块定义)规范,就是这样需求背景下的产物。

然而这些社区提出的规范终究只是为了解决一时的需求,随着历史的发展,新的模块化规范不断涌入、消亡。

直到ESM规范被提出。

ESM规范是ES标准的模块化规范,他的早期讨论可以追溯到2009年。

你可以在这里看到ESM规范的历史es-module-history[3]

ESM将模块规范分为三个阶段:

  • 模块加载 --> 模块实例化 --> 模块执行

其中「模块加载」由宿主环境提供的loader完成(比如在浏览器环境,loader的行为由HTML规范[4]定义)。

「模块实例化」与「模块执行」由ESM规范定义执行流程。

区别于CJS规范的同步执行,ESM规范将流程拆解为3个独立阶段。

「模块加载」同步、异步与否由宿主环境决定。

支持不同宿主环境,抹平多端差异、能力比其他规范都强大(后文会介绍)、再加上血统纯正(ES官方提出),

使得ESM规范一统前端「看似」指日可待。

然而,此时社区已经有大量基于CJS规范产出的开源包、组件,他们无法立刻切换到ESM规范。

所以,JS生态的现状是:会处于、并将长期处于CJS规范的库与ESM规范的库共存的状态。

但是最终,ESM规范一定会成为主导,毕竟他的优点太多(同样,后文会介绍)。

规范割裂带来的机会

js中ESM规范的示例分析

当前模块化规范的混乱,对开源大佬们来说,就是机会。

为了让开发者将更多精力放在业务,而不是模块规范的适配上。

很多开源「工具集」都试图抹平模块化差异,比如:

  • 在babel中使用babel-plugin-transfORM-commonjs可以将CJS规范的代码转换为ESM规范

  • 为了一刀切解决当前ESM、CJS、浏览器script标签导入这3种规范互相不兼容的情况,提出了兼容三者格式的UMD(Universal Module  Definition)规范

一些「工具集」利用模块化规范的不同与其他竞品形成差异化竞争,比如:

  • browserify这款打包工具的卖点是:使用CJS规范打包,使一份代码同时在Node环境与浏览器环境(打包后)执行。

js中ESM规范的示例分析

其中,在浏览器环境中,Node的一些核心库(如events、stream、path...)会被打包成浏览器支持的版本。

  • Vite在DEV环境使用ESM规范构建模块间的依赖关系。

依赖于大部分现代浏览器原生支持ESM规范,省去了打包的过程,使其编译速度大大提升。

  • rollup原生对ESM提供更多支持。

严格支持ESM规范,并提供更好的静态分析,使rollup一度提供性能更优异的treeShaking能力。

成为更多库打包工具的首选。

与webpack这样的的大而全方案形成差异竞争。

规范割裂带来的痛

可以看到,由于底层宿主环境对模块化规范支持的割裂,需要上层工具集来抹平模块规范的差异。

设想一个同时使用了webpack、babel、TS的项目。

这3个工具集都对多种模块规范有兼容处理。比如:

单独使用babel时,对于如下代码:

import a from 'lib'; console.log(a);

会被babel编译为:

"use strict";  var _lib = _interopRequireDefault(require("lib"));  function _interopRequireDefault(obj) {   return obj && obj.__esModule ? obj : { default: obj }; }  console.log(_lib.default);

ESM的「默认导出」会被编译为包含default属性的对象。

你可以打开babel playground[5]试试

当多个「工具集」在同一个项目中,为了各自目的做着同一件事(抹平模块化规范差异),

一旦工具链中某个插件配置有一丝丝不符合预期,或者引入了一个不符合预期的包,那么艰难的debug就此开始了......

js中ESM规范的示例分析

曙光即使当前有诸多不便,历史的进程是无法阻止的,那些被历史巨轮甩下并碾碎的模块化规范,会逐渐消失在开发者的视野中。

看完上述内容,你们掌握js中ESM规范的示例分析的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注编程网node.js频道,感谢各位的阅读!

--结束END--

本文标题: js中ESM规范的示例分析

本文链接: http://www.lsjlt.com/news/81705.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • js中ESM规范的示例分析
    js中ESM规范的示例分析,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。前端发展到如今,社区生态已经非常丰富。在无数开源大神的努力下,很多前端...
    99+
    2024-04-02
  • js模块化规范的示例分析
    这篇文章主要介绍js模块化规范的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1. CommonJS用于服务端模块化编程,比如nodejs就采用此规范;一个文件就是一个模块,...
    99+
    2024-04-02
  • JS前端模块化规范的示例分析
    这篇文章将为大家详细讲解有关JS前端模块化规范的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。Script 标签其实最原始的 JavaScript 文件加载方式,...
    99+
    2024-04-02
  • Nodejs中模块规范的示例分析
    这篇文章主要为大家展示了“Nodejs中模块规范的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Nodejs中模块规范的示例分析”这篇文章吧。CommonJS 模块规范以前加载 JS 文...
    99+
    2023-06-15
  • css中BEM书写规范的示例分析
    这篇文章给大家分享的是有关css中BEM书写规范的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。BEM是基于组件的web开发方法。其思想是将用户界面分隔为独立的块,从而使开发复杂的UI界面变得更简单和快,...
    99+
    2023-06-08
  • CSS规范BEMCSS和OOCSS的示例分析
    这篇文章主要介绍CSS规范BEMCSS和OOCSS的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!BEM(Block, Element, Modifier) CSS 介绍BEM是一种 前端命名规范 ,顾名思议...
    99+
    2023-06-08
  • web前端开发规范的示例分析
    小编给大家分享一下web前端开发规范的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!规范说明此为前端开发团队遵循和约定的...
    99+
    2024-04-02
  • Vue中编码技巧与规范的示例分析
    这篇文章主要介绍Vue中编码技巧与规范的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!实例1. 使用对象代替 if 及 switch在很多情况下,我们经常会遇到循环判断执行赋...
    99+
    2024-04-02
  • unity程序初级规范的示例分析
    这篇文章主要为大家展示了“unity程序初级规范的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“unity程序初级规范的示例分析”这篇文章吧。1 代码不能用拼音,缩写2&nbs...
    99+
    2023-06-03
  • Java语言编码规范的示例分析
    这篇文章跟大家分析一下“Java语言编码规范的示例分析”。内容详细易懂,对“Java语言编码规范的示例分析”感兴趣的朋友可以跟着小编的思路慢慢深入来阅读一下,希望阅读后能够对大家有所帮助。下面跟着小编一起深入学习“Java语言编码规范的示例...
    99+
    2023-06-03
  • Java代码书写规范的示例分析
    这篇文章主要介绍了Java代码书写规范的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、目的   对于代码,首要要求是它必须正确,能够按照程序员的真实思想...
    99+
    2023-06-03
  • CSS和Sass开发规范的示例分析
    这篇文章将为大家详细讲解有关CSS和Sass开发规范的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。ID and class naming ID和class(类)...
    99+
    2024-04-02
  • js中try、catch、finally执行规则的示例分析
    这篇文章将为大家详细讲解有关js中try、catch、finally执行规则的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。try:  语句测试代码块的...
    99+
    2024-04-02
  • js的一些潜在规则示例分析
    目录宏任务和微任务语句的执行过程 (Completion Record )文法词法语句是否需要加分号no LineTerminator here规则脚本和模块声明提升解析HTMLDO...
    99+
    2023-02-21
    js潜在规则 js 规则
  • CSS书写规范和顺序的的示例分析
    这篇文章将为大家详细讲解有关CSS书写规范和顺序的的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。CSS书写顺序 1.位置属性(position, to...
    99+
    2024-04-02
  • js中module的示例分析
    这篇文章给大家分享的是有关js中module的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。JS本身是一个多才多艺的语言,一个可以用自己编译自己的自由度极高的语言。正因为...
    99+
    2024-04-02
  • js中ParseInt()的示例分析
    这篇文章主要介绍了js中ParseInt()的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。parseInt()是内置的 JS 函数...
    99+
    2024-04-02
  • CSS中BEM命名规范实例分析
    这篇文章主要讲解了“CSS中BEM命名规范实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“CSS中BEM命名规范实例分析”吧!   一什么是BEM命...
    99+
    2024-04-02
  • Javascript中DOM范围的示例分析
    这篇文章主要介绍Javascript中DOM范围的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!创建范围Document类型中定义了createRange()方法。在兼容DO...
    99+
    2024-04-02
  • js中继承的示例分析
    这篇文章给大家分享的是有关js中继承的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。继承的简介继承”是JavaScript面向对象设计的重要一环,愿你认真读完本文,吃透继...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作