美文网首页大前端从入门到跑路
在Nodejs或浏览器运行ESM代码

在Nodejs或浏览器运行ESM代码

作者: brandonxiang | 来源:发表于2021-09-05 17:51 被阅读0次

前言

经典面试题目就是《Common.js和ES module的区别》,这一题很多人都会熟练地背出答案。

Commonjs

  • CommonJs可以动态加载语句,代码发生在运行时
  • CommonJs导出值是拷贝, 不好排查引起变量污染

ES module(后续简称esm)

esm是JavaScript模块化的未来。因为它解决了变量污染,代码维护,代码依赖的问题。它让你的代码更加科学。这也是deno默认采用esm的原因。

回归正题,我们有什么方法在Nodejs或者浏览器直接运行esm代码,这是个有趣而又实际的问题。

如何在Nodejs环境允许

1.利用编译工具运行esm

最常见的方式是利用webpack等打包工具搭配babel使用。随着webpack和vue的大热,这些工具似乎成为了标配,但是webpack的缺点也很明显,它能让commonjs和esm的混写,导致代码存在一些写法不规范的情况,我相信这种情况是普遍出现在业务代码里面,也存在于antd3这样的知名第三方组件库中。

rollup则是基于ES6的语法规范进行编译,它的轻便小巧,非常适合npm库的打包。新兴的打包工具例如esbuildswc,也可以实现编译打包,即使速度越来越快,但是还是需要编译的过程。这些仓库很重要的一个特点就是使用esm语法。

以上这些工具都可以应用于esm语法编译,但是有很多项目不一定需要打包编译这样耗时的流程的,例如一些cli工具、简易微服务等,如何保证高效正确的运行esm代码呢?

2. 利用第三方库运行esm

在Nodejs版本较低的情况,我们可以利用一些工具,工具的使用形式有几种,一种是Module Loader,另一种是Command Line(简称为cli)。

Module Loader,这里介绍standard-things/esm,它可以preload第三方提供的esm包,从此,可以做到babelless, bundleless。你不需要使用大型编译工具也可以直接运行esm代码,使用方式如下。

node -r esm index.js

同样,egoist/esbuild-register这个库在esbuild的支持下,同样可以做到Module Loader的效果,利用esbuild的高性能特性,代码运行效率更高。

node -r esbuild-register index.js

Command Line,基于封装后的cli,不过是换一种形式进行模块的提前处理。babel-node直接利用它的babel语法优势来运行esm代码。由于babel本身还是js的实现,它的官方文档也表明了不建议在生产环境使用,会导致内存高占用的问题,这也是这一类工具的通病。

babel-node index.js

同样,esno可以直接在命令行运行esm代码。原理基于esbuild。在这里更推荐使用这种方式,鉴于esbuild是由go语言实现,能够较大程度解决内存高占用的问题,保证了一定的执行性能。

esno index.ts
esmo index.ts

这一类第三方仓库适合在低版本nodejs且非生产环境使用,它们的存在是为了便利性,而并非实用性和稳定性。怎么样才能高效地运行esm代码?

3.Native Nodejs运行esm

Node verison 13.2.0 起开始正式支持 ES Modules 特性

所以利用Native Nodejs环境运行esm代码是非常必要的,高版本的Nodejs提供了直接运行esm的功能,这里建议使用lts14版本。有两种方式运行esm代码:

第一种,package.json中填写type: "modules",表明模块的类型。此后,直接运行node index.js即可。

// pakage.json
{
  ...
  "type": "modules"
}

第二种,则是将文件名改成.mjs,标明该文件是esm代码。这两种方式最大的区别则是模块作用域。前者是包的作用域,它的声明是以package为维度。后者则是以文件为维度,不受限于包的作用域。

如何在浏览器运行esm

浏览器script type="module"兼容性

浏览器的情况有别于Nodejs环境,在大部分的新版本浏览器都支持esm的运行。esm 级别的代码编译和打包,可以有效地减少包的体积和资源传输速度。这也是为什么像 vite 这样的框架会采用现代浏览器的打包模式(外加legacy兼容模式)的原因。具体的原理是在 html 当中的 script 标签加入type="module" 则表明它引入的是esm代码,当旧浏览器没法支持esm的情况下,它会读取nomodule script中的地址,读取兼容版本的 js 代码。这样一来,可以有效地减少大部分浏览器加载的 js 体积,又保证了老浏览器的兼容性问题。

<script type="module" src="dist/index.js"></script>
<script nomodule src="dist/index.legacy.js"></script>

总结

如今Nodejs和浏览器环境都能对esm语法有了很好的 Native 支持。作为前端工程师的我们,应该要保持着技术的前瞻性,在写一个仓库的时候,我们要想到要用typescript,esm还是common.js呢?为什么我们不选择比较新的 js 运行环境,迎接Javascript的第三个时代,参考《ESM Import与Bundleless》

参考资料

2020年我们可以在Node中使用ES Modules了吗

相关文章

  • 在Nodejs或浏览器运行ESM代码

    前言 经典面试题目就是《Common.js和ES module的区别》,这一题很多人都会熟练地背出答案。 Comm...

  • vscode 单步调试 nodejs 代码

    纯nodejs的后台代码(即,不通过浏览器运行,仅跑在命令行上的nodejs程序),可以通过vscode的debu...

  • 如何创建一个自己的博客

    1.首先下载安装nodejs Nodejs是一个运行环境,这个环境运行js代码(作用相当于浏览器),大家可搜索ht...

  • ES6快速学习(一)变量声明

    ES6 代码在浏览器环境运行!代码在浏览器环境运行!代码在浏览器环境运行!我用的google!!! 变量声明 le...

  • 关于事件循环

    首先,js 是单线程的语言,eventloop是js的执行机制,在不同的运行环境(浏览器或Nodejs)下,执行顺...

  • python开启局域网共享文件-8080端口

    很简单的小玩意 代码 效果 运行上部分代码后,在浏览器输入http://localhost:8080,或本机ip+...

  • 下载文件中文名乱码

    原来NodeJS代码: 问题:下载的文件中文名在chrome浏览器正常,在firefox浏览器下不正常修改代码如下...

  • JS执行环境

    JS脚本可在 浏览器和NodeJS中运行 浏览器 新建html文件 ...

  • nodejs01

    nodejs介绍 nodejs是基于chrome V8浏览器的运行环境 没有界面操作,只是ECMAscript 没...

  • Web前端的前世今生

    什么是前端 前端:针对浏览器的开发,代码在浏览器运行 后端:针对服务器的开发,代码在服务器运行 前端的前世 在互联...

网友评论

    本文标题:在Nodejs或浏览器运行ESM代码

    本文链接:https://www.haomeiwen.com/subject/mmyubltx.html