Nodejs14大版本中新增特性总结

 Node.js 是一个基于 Chrome V8 引擎 的 JavaScript 运行时。在 2020 年 10 月 27 日 Node.js v14.15.0 LTS 版已发布,即长期支持版本,其中包含了很多很棒的新功能,以下内容也是基于笔者在日常 Node.js 工作和学习中所总结的,可能不全,同时也欢迎补充,有些功能之前也曾单独写过文章来介绍,接下让我们一起看看都有哪些新的变化?

创新互联主营乳山网站建设的网络公司,主营网站建设方案,重庆APP开发,乳山h5微信小程序开发搭建,乳山网站营销推广欢迎乳山等地区企业咨询

Optional Chaining(可选链)

如果我们使用 JavaScript 不管是用在前端或者 Node.js 服务端都会出现如下情况,因为我们有时是不确定 user 对象是否存在,又或者 user 对象里面的 address 是否存在,如果不这样判断, 可能会得到类似于 Cannot read property 'xxx' of undefined 这样的类似错误。

 
 
 
 
  1. const user = { 
  2.   name: 'Tom', 
  3.   address: { 
  4.     city: 'ZhengZhou' 
  5.   } 
  6. if (user && user.address) { 
  7.   console.log(user.address.city) 
  8. }

现在我们有一种优雅的写法 "可选链操作符",不必明确的验证链中的每个引用是否有效,以符号 "?." 表示,在引用为 null 或 undefined 时不会报错,会发生短路返回 undefined。

 
 
 
 
  1. user.address?.city 
  2. user.address?.city?.length 
  3. // 结合 ?.[] 的方式访问相当于 user.address['city'] 
  4. user.address?.['city'] 
  5. // 结合 delete 语句使用,仅在 user.address.city 存在才删除 
  6. delete user.address?.city

参考 https://v8.dev/features/optional-chaining

Nullish Coalescing(空值合并)

逻辑或操作符(||)会在左侧为假值时返回右侧的操作符,例如我们传入一个属性为 enabled:0 我们期望输出左侧的值,则是不行的。

 
 
 
 
  1. function Component(props) { 
  2.   const enable = props.enabled || true; // true 
  3. Component({ enabled: 0 })

现在我们可以使用 **空值合并操作符(??)**来实现,仅当左侧为 undefined 或 null 时才返回右侧的值。

 
 
 
 
  1. function Component(props) { 
  2.   const enable = props.enabled ?? true; // 0 
  3. Component({ enabled: 0 })

参考:https://v8.dev/features/nullish-coalescing

Intl.DisplayNames(国际化显示名称)

对于国际化应用需要用到的语言、区域、货币、脚本的名称,现在 JavaScript 开发者可以使用 Intl.DisplayNames API 直接访问这些翻译,使应用程序更轻松的显示本地化名称。

Language(语言)

 
 
 
 
  1. let longLanguageNames = new Intl.DisplayNames(['zh-CN'], { type: 'language' }); 
  2. longLanguageNames.of('en-US'); // 美国英语 
  3. longLanguageNames.of('zh-CN'); // 中文(中国) 
  4. longLanguageNames = new Intl.DisplayNames(['en'], { type: 'language' }); 
  5. longLanguageNames.of('en-US'); // American English 
  6. longLanguageNames.of('zh-CN'); // Chinese (China)

Region(区域)

 
 
 
 
  1. let regionNames = new Intl.DisplayNames(['zh-CN'], {type: 'region'}); 
  2. regionNames.of('US'); // 美国 
  3. regionNames.of('419'); // 拉丁美洲 
  4. regionNames = new Intl.DisplayNames(['en'], {type: 'region'}); 
  5. regionNames.of('US'); // United States 
  6. regionNames.of('419'); // Latin America

Currency(货币)

 
 
 
 
  1. let currencyNames = new Intl.DisplayNames(['zh-CN'], {type: 'currency'}); 
  2. currencyNames.of('CNY'); // 人民币 
  3. currencyNames.of('USD'); // 美元 
  4. currencyNames = new Intl.DisplayNames(['en'], {type: 'currency'}); 
  5. currencyNames.of('CNY'); // Chinese Yuan 
  6. currencyNames.of('USD'); // US Dollar

Script(脚本)

 
 
 
 
  1. let scriptNames = new Intl.DisplayNames(['zh-CN'], {type: 'script'}); 
  2. scriptNames.of('Hans'); // 简体 
  3. scriptNames.of('Latn'); // 拉丁文 
  4. scriptNames = new Intl.DisplayNames(['en'], {type: 'script'}); 
  5. scriptNames.of('Hans'); // Simplified
  6. scriptNames.of('Latn'); // Latin

参考:https://v8.dev/features/intl-displaynames

上述实例用到的国家代号和 code 都可从参考地址获取。

Intl.DateTimeFormat(国际化处理日期时间格式)

Intl.DateTimeFormat API 用来处理特定语言环境的日期格式。

 
 
 
 
  1. const date = new Date(); 
  2. // Sunday, January 10, 2021 at 9:02:29 PM GMT+8 
  3. new Intl.DateTimeFormat('en-US', { dateStyle: 'full', timeStyle: 'long'}).format(date) 
  4. // 21/1/10 中国标准时间 下午9:02:29.315 
  5. new Intl.DateTimeFormat('zh-CN', { 
  6.   year: '2-digit', 
  7.   month: 'numeric', 
  8.   day: 'numeric', 
  9.   hour: 'numeric', 
  10.   minute: 'numeric', 
  11.   second: 'numeric', 
  12.   fractionalSecondDigits: 3, 
  13.   timeZoneName: 'long' 
  14. }).format(date)

参考:

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Intl/DateTimeFormat/DateTimeFormat

String.prototype.matchAll (throws on non-global regex)

matchAll() 返回一个包含所有匹配正则表达式的结果,返回值为一个不可重用(不可重用意思为读取完之后需要再次获取)的迭代器。

matchAll() 方法在 Node.js v12.4.0 以上版本已支持,该方法有个限制,如果设置的正则表达式没有包含全局模式 g ,在 Node.js v14.5.0 之后的版本如果没有提供会抛出一个 TypeError 异常。

 
 
 
 
  1. // const regexp = RegExp('foo[a-z]*','g'); // 正确 
  2. const regexp = RegExp('foo[a-z]*'); // 错误,没有加全局模式 
  3. const str = 'table football, foosball, fo'; 
  4. const matches = str.matchAll(regexp); // TypeError: String.prototype.matchAll called with a non-global RegExp argument 
  5. for (const item of matches) { 
  6.   console.log(item); 
  7. }

参考:

https://node.green/#ES2020-features-String-prototype-matchAll-throws-on-non-global-regex

Async Local Storage(异步本地存储)

Node.js Async Hooks 模块提供了 API 用来追踪 Node.js 程序中异步资源的声明周期,在最新的 v14.x LTS 版本中新增加了一个 AsyncLocalStorage 类可以方便实现上下文本地存储,在异步调用之间共享数据,对于实现日志链路追踪场景很有用。

下面是一个 HTTP 请求的简单示例,模拟了异步处理,并且在日志输出时去追踪存储的 id。

 
 
 
 
  1. const http = require('http'); 
  2. const { AsyncLocalStorage } = require('async_hooks'); 
  3. const asyncLocalStorage = new AsyncLocalStorage(); 
  4. function logWithId(msg) { 
  5.   const id = asyncLocalStorage.getStore(); 
  6.   console.log(`${id !== undefined ? id : '-'}:`, msg); 
  7. let idSeq = 0; 
  8. http.createServer((req, res) => { 
  9.   asyncLocalStorage.run(idSeq++, () => { 
  10.     logWithId('start'); 
  11.     setImmediate(() => { 
  12.       logWithId('processing...'); 
  13.       setTimeout(() => { 
  14.         logWithId('finish'); 
  15.         res.end(); 
  16.       }, 2000) 
  17.     }); 
  18.   }); 
  19. }).listen(8080);

下面是运行结果,我在第一次调用之后直接调用了第二次,可以看到我们存储的 id 信息与我们的日志一起成功的打印了出来。

image.png

便利性的同时也会牺牲一些性能上的代价。

ES Modules 支持

ES Modules 的支持总体上来说是个好事,进一步的规范了 Node.js 与浏览器的模块生态,使之进一步趋同,同时避免了进一步的分裂。

在当前 Node.js v14.x LTS 版本中已移除试验性支持,现在使用无需使用标志了,它使用 import、export 关键字,两种使用方式:

使用 .mjs 扩展名

 
 
 
 
  1. // caculator.mjs 
  2. export function add (a, b) { 
  3.   return a + b; 
  4. }; 
  5. // index.mjs 
  6. import { add } from './caculator.js'; 
  7. console.log(add(4, 2)); // 6

告诉 Node.js 将 JavaScript 代码视为 ES Modules

默认情况下 Node.js 将 JavaScript 代码视为 CommonJS 规范,所以我们要在上面使用扩展名为 .mjs 的方式来声明,除此之外我们还可以在 package.json 文件中 设置 type 字段为 module 或在运行 node 时加上标志 --input-type=module 告诉 Node.js 将 JavaScript 代码视为 ES Modules。

 
 
 
 
  1. // package.json 
  2.   "name": "esm-project", 
  3.   "type": "module", 
  4.   ... 
  5. }

前端的同学可能会对以上使用 ES Modules 的方式很熟悉。

Top-Level Await(顶级 await 支持)

顶级 await 支持在异步函数之外使用 await 关键字,在 Node.js v14.x LTS 版本中已去掉试验性支持,现在使用也不再需要设置标志。

 
 
 
 
  1. import fetch from 'node-fetch'; 
  2. const res = await fetch(url)

也可以像调用函数一样动态的导入模块。

 
 
 
 
  1. const myModule = await import('./my-module.js');

对于异步资源,之前我们必须在 async 函数内才可使用 await,这对一些在文件顶部需要实例化的资源可能会不 好操作,现在有了顶级 await 我们可以方便的在文件顶部对这些异步资源做一些初始化操作。

Diagnostic report(诊断报告)

Diagnostic report 是 Node.js v14.x LTS 提供的一个稳定功能,在某些情况下会生成一个 JSON 格式的诊断报告,可用于开发、测试、生产环境。报告会提供有价值的信息,包括:JavaScript 和本机堆栈信息、堆统计信息、平台信息、资源使用情况等,帮助用户快速追踪问题。

https://github.com/IBM/report-toolkit 是 IBM 开发的一个款工具,用于简化报告工具的使用,如下是一个简单 Demo 它会造成服务的内存泄漏。

 
 
 
 
  1. const total = []; 
  2. setInterval(function() { 
  3.   total.push(new Array(20 * 1024 * 1024)); // 大内存占用,不会被释放 
  4. }, 1000)

最终生成的 JSON 报告被 report-toolkit 工具诊断的结果可能是下面这样的。

Stream

新版本中包含了对 Stream 的一些更改,旨在提高 Stream API 的一致性,以消除歧义并简化 Node.js 核心各个部分的行为,例如:

  •  http.OutgoingMessage 与 stream.Writable 类似
  •  net.Socket 的行为与 stream.Duplex 完全相同
  •  一个显著的变化 autoDestroy 的默认值为 true,使流在结束之后始终调用 _destroy

参考:

https://nodejs.medium.com/node-js-version-14-available-now-8170d384567e

使用异步迭代器

使用异步迭代器我们可以对 Node.js 中的事件、Stream 亦或者 MongoDB 返回数据遍历,这是一件很有意思的事情,尽管它不是 Node.js v14.x 中新提出的功能,例如 event.on 是在 Node.js v12.16.0 才支持的,这些目前看到的介绍还不太多,因此我想在这里做下简单介绍。

在 Events 中使用

Node.js v12.16.0 中新增了 events.on(emitter, eventName) 方法,返回一个迭代 eventName 事件的异步迭代器,例如启动一个 Node.js 服务可以如下这样写,想知道它的原理的可以看笔者下面提到的相关文章介绍。

 
 
 
 
  1. import { createServer as server } from 'http'; 
  2. import { on } from 'events'; 
  3. const ee = on(server().listen(3000), 'request'); 
  4. for await (const [{ url }, res] of ee) 
  5.   if (url === '/hello') 
  6.     res.end('Hello Node.js!'); 
  7.   else 
  8.     res.end('OK!');

在 Stream 中使用

以往我们可以通过 on('data') 以事件监听的方式读取数据,通过异步迭代器可以一种更简单的方式实现。

 
 
 
 
  1. async function readText(readable) { 
  2.   let data = ''; 
  3.   for await (const chunk of readable) { 
  4.     data += chunk; 
  5.   } 
  6.   return data; 
  7. }

目前在 JavaScript 中还没有被默认设定 [Symbol.asyncIterator] 属性的内建对象,在 Node.js 的一些模块 Events、Stream 中是可使用的,另外你还可以用它来遍历 MongoDB 的返回结果。

网页标题:Nodejs14大版本中新增特性总结
网页路径:http://www.csdahua.cn/qtweb/news13/294213.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网