ES2020 已定稿,真实场景案例分析


Posted in Javascript onMay 25, 2020

近年来,JavaScript 的发展非常迅速。 尤其是在2015 年 ES6 发布之后,情况变得更好。

现在 许多新的特性被提议包括在 ES2020版本中。好消息是这些已经已经敲定。 现在,我们获得了最终定稿的功能清单,它们将在被批准发布之后出现在备受期待的 ES2020 中。 其中一些功能使我非常兴奋,因为在它们存在之前编写代码时遇到将会遇到很多麻烦。 让我们看看它们是什么吧!

可选链操作符(Optional Chaining Operator)

对我个人来说,这是 ES2020最令人兴奋的特点之一。 我已经编写了很多程序,这些程序将会从这个新特性中获益匪浅。

可选链操作符允许您安全地访问对象的深嵌套属性,而不必检查每个属性是否存在。 让我们看看这个特性对我们有什么帮助。

拥有可选链操作符之前

const user = {
 firstName: "Joseph",
 lastName: "Kuruvilla",
 age: 38,
 address: {
 number: "239",
 street: "Ludwig Lane",
 city: "Chennai",
 zip: "600028",
 prop1: {
  prop2: {
  prop3: {
   prop4: {
   value: "sample",
   },
  },
  },
 },
 },
};

if (user && user.address) {
 console.log(user.address.zip);
 //600028
}

if (
 user &&
 user.address &&
 user.address.prop1 &&
 user.address.prop1.prop2 &&
 user.address.prop1.prop2.prop3 &&
 user.address.prop1.prop2.prop3.prop4
) {
 console.log(user.address.prop1.prop2.prop3.prop4.value);
 //sample
}

//Accessing unexisting property
console.log(user.address.prop102.po);
//Error

正如您在上面看到的,您必须检查每个级别中是否存在该属性,以避免出现无法读取未定义属性“ po”的错误。 随着嵌套级别的增加,手动检查的属性数量也会增加。 这意味着我们必须检查每个级别,以确保它不会在遇到未定义或空对象时崩溃。

拥有可选链式操作符之后

随着可选链式操作符 (Optional Chaining) 的引入,我们前端的工作变得容易多了。 通过简单地使用可选链式操作符?.我们可以访问深嵌套的对象,而不必检查未定义或空对象。

让我们看看它是如何运作的。

const user = {
 firstName: "Joseph",
 lastName: "Kuruvilla",
 age: 38,
 address: {
 number: "239",
 street: "Ludwig Lane",
 city: "Chennai",
 zip: "600028",
 prop1: {
  prop2: {
  prop3: {
   prop4: {
   value: "sample",
   },
  },
  },
 },
 },
};

console.log(user?.address?.zip);
// 600028

console.log(user?.address?.prop1?.prop2?.prop3?.prop4?.value);
// sample

//Accessing unexisting property
console.log(user?.address?.prop102?.po);
//undefined

太神奇了! ES2020 成功地通过引入一个单独的代码操作符?.来减少了如此多的代码行数!

空值合并操作符(Nullish coalescing operator)

这是另一个令我兴奋的功能,当我第一次在proposal stage, 了解到的时候,我由衷的喜欢这个特性,因为我已经历了编写单独的函数来手动检查这个特性的麻烦。

空值合并操作符允许您检查nullish值而不是  falsey值。 Nullish 值是指  nullundefined的值。 而 falsey 值是诸如空字符串、数字0、  undefined、  null、  false、  NaN等等的值。 这对你来说可能听起来没什么不同,但是在现实中,这意味着很多。

让我们看看这是怎么回事。

在有空值合并操作符之前

我最近做了一个项目,我需要允许黑暗模式(Dark Mode)切换功能。 我必须检查输入是true还是  false。 如果用户没有设置任何值,则默认为  true。 下面就是我如何在有空值合并操作符之前实现它的:

const darkModePreference1 = true;
const darkModePreference2 = false;
const darkModePreference3 = undefined;
const darkModePreference4 = null;
const getUserDarkModePreference = (darkModePreference) => {
 if (darkModePreference || darkModePreference === false) {
 return darkModePreference;
 }
 return true;
};
getUserDarkModePreference(darkModePreference1);
// true
getUserDarkModePreference(darkModePreference2);
// false
getUserDarkModePreference(darkModePreference3);
// true
getUserDarkModePreference(darkModePreference4);
// true

在有空值合并操作符之后

在有空值合并操作符之后,您所要做的就是使用??操作符。不需要  if语句:

const darkModePreference1 = true;
const darkModePreference2 = false;
const darkModePreference3 = undefined;
const darkModePreference4 = null;
const getUserDarkModePreference = (darkModePreference) => {
 return darkModePreference ?? true;
};
getUserDarkModePreference(darkModePreference1);
// true
getUserDarkModePreference(darkModePreference2);
// false
getUserDarkModePreference(darkModePreference3);
// true
getUserDarkModePreference(darkModePreference4);
// true

这里基本上发生的情况是,如果变量 darkModePreference 包含一个nullish值,那么将值 true 赋给它。 简单,简短,易于理解。

动态 import(Dynamic Imports)

这个特性将帮助您的应用程序更加高效的执行, 动态 import 允许您将 JS 文件作为原生应用用程序中的模块动态导入。 在 ES2020之前,不管是否使用模块,都应该导入模块。

例如,假设我们需要添加一个功能来下载 pdf 格式的文件。

让我们看看如何在 动态 import 之前和之后实现这一点。

在 动态 import  之前

实际上,不会所有的页面访问者使用下载 pdf 的选项。 但是,无论我们的访客是否使用它,它仍然需要被导入。 这意味着这个 pdf 模块也可以在页面加载期间被下载。

import { exportAsPdf } from "./export-as-pdf.js";
const exportPdfButton = document.querySelector(".exportPdfBtn");
exportPdfButton.addEventListener("click", exportAsPdf);

这种开销可以通过使用延迟加载模块(lazy loaded modules)来减少。 可以通过称为代码分割 (code-splitting)的方法来实现,这在 Webpack 或其他模块打包工具已经可以使用了。但是对于 ES2020,我们可以直接使用它了,而不需要模块打包工具,如 Webpack。

在动态导入(动态 import)之后

const exportPdfButton = document.querySelector('.exportPdfBtn');
exportPdfButton.addEventListener('click', () => {
 import('./export-as-pdf.js')
 .then(module => {
  module.exportAsPdf()
 })
 .catch(err => {
  // handle the error if the module fails to load
 })
})

正如您在上面的代码中看到的,现在只有在需要模块时才延迟加载模块。 从而减少开销和页面加载时间。

Promise.allSettled

如果你有一个场景,在所有 Promise 都完成之后必须执行一个任务,那么你可能使用Promise.all()方法。 但是这个方法有一个缺点。 当你的任何一个 Promise 被 Rejected 时,Promise 方法就会抛出一个错误。 这意味着您的代码不会等到所有的 Promise 都完成。

这可能不是你想要的。 如果你想要这样的东西: “我不在乎他们的结果。 只需全部运行” ,那么你可以使用新的Promise.allSettled()方法。 这种方法只有在你的所有 Promise 都  settled?? 要么  Resolved,要么  Rejected?? 时才会  Resolved

在拥有 Promise.allSettled 之前

const PromiseArray = [
 Promise.resolve(100),
 Promise.reject(null),
 Promise.resolve("Data release"),
 Promise.reject(new Error("Something went wrong")),
];
Promise.all(PromiseArray)
 .then((data) =>
 console.log("all resolved! here are the resolve values:", data)
 )
 .catch((err) => console.log("got rejected! reason:", err));
//got rejected! reason: null

如上所述,当其中一个 Promise 被rejected时, Promise 就会抛出错误。

在拥有 Promise.allSettled 之后

const PromiseArray = [
 Promise.resolve(100),
 Promise.reject(null),
 Promise.resolve("Data release"),
 Promise.reject(new Error("Something went wrong")),
];
Promise.allSettled(PromiseArray)
 .then((res) => {
 console.log(res);
 })
 .catch((err) => console.log(err));
//[
// {status: "fulfilled", value: 100},
// {status: "rejected", reason: null},
// {status: "fulfilled", value: "Data release"},
// {status: "rejected", reason: Error: Something went wrong ...}
//]

尽管有些 Promise 被rejected了,Promise.allSettled 返回了所有的 Promise 的结果。

globalThis

globalThis包含对全局对象的引用,与环境无关。 在浏览器中,全局对象是  window对象。 在 Node 环境中,全局对象是  global或者 Web workers 中的  self

在拥有 globalThis 之前

我们在工作中会有需要编写一份同时运行在 Node 和浏览器中的通用代码,当我们要取得全局对象时,通常需要做很多工作和逻辑判断:

beforeGlobalThis = (typeof window !== "undefined"
? window
: (typeof process === 'object' &&
 typeof require === 'function' &&
 typeof global === 'object')
 ? global
 : this);

beforeGlobalThis.tuture = '小若燕雀,亦可一展宏图';

在拥有 globalThis 之后

我们可以直接使用globalThis去引用全局对象,而不用去担心环境的问题:

globalThis.tuture = '小若燕雀,亦可一展宏图';

上面的代码在浏览器或者 Node 环境中都是通用的,你可以放心使用!

BigInt

允许您使用大于 Javascript 中允许的最大值的数字。 这个数字是pow(2,53)-1。 尽管这不能向后兼容,因为传统的数字系统(IEEE 754)不能支持这种大小的数字。

String.matchall

matchAll()是一个与正则表达式相关的方法。 此方法返回与正则表达式匹配的字符串的所有结果的迭代器,包括捕获组。 这个方法已经被添加到 String 原型中。

参考资源

ECMA
InfoQ
Article by Tyler Hawkins

总结

到此这篇关于ES2020 已定稿,真实场景案例分析的文章就介绍到这了,更多相关ES2020 已定稿内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Javascript 相关文章推荐
javascript 读取图片文件的大小
Jun 25 Javascript
functional继承模式 摘自javascript:the good parts
Jun 20 Javascript
js获取url中"?"后面的字串方法
May 15 Javascript
使表格的标题列可左右拉伸jquery插件封装
Nov 24 Javascript
jQuery插件jquery.kxbdmarquee.js实现无缝滚动效果
Feb 15 Javascript
Vue关于数据绑定出错解决办法
May 15 Javascript
JavaScript中最常用的10种代码简写技巧总结
Jun 28 Javascript
JavaScript实现焦点进入文本框内关闭输入法的核心代码
Sep 20 Javascript
详解如何使用PM2将Node.js的集群变得更加容易
Nov 15 Javascript
浅析node Async异步处理模块用例分析及常用方法介绍
Nov 17 Javascript
微信小程序实现的点击按钮 弹出底部上拉菜单功能示例
Dec 20 Javascript
AJAX在JQuery中的应用详解
Jan 30 jQuery
Javascript原型链及instanceof原理详解
May 25 #Javascript
Node登录权限验证token验证实现的方法示例
May 25 #Javascript
详解Node.js使用token进行认证的简单示例
May 25 #Javascript
基于redis的小程序登录实现方法流程分析
May 25 #Javascript
JSONP解决JS跨域问题的实现
May 25 #Javascript
JS实现时间校验的代码
May 25 #Javascript
使用Typescript和ES模块发布Node模块的方法
May 25 #Javascript
You might like
PHP文件去掉PHP注释空格的函数分析(PHP代码压缩)
2013/07/02 PHP
PHP 面向对象程序设计(oop)学习笔记 (四) - 异常处理类Exception
2014/06/12 PHP
PHP的fsockopen、pfsockopen函数被主机商禁用的解决办法
2014/07/08 PHP
PHP面向对象程序设计之接口用法
2014/08/20 PHP
PHP将进程作为守护进程的方法
2015/03/19 PHP
Thinkphp5 自定义上传文件名的实现方法
2019/07/23 PHP
JavaScript初学者需要了解10个小技巧
2010/08/25 Javascript
让网页跳转到指定位置的jquery代码非书签
2013/09/06 Javascript
ActiveX控件与Javascript之间的交互示例
2014/06/04 Javascript
jQuery获取节点和子节点文本的方法
2014/07/22 Javascript
基于jquery实现一个滚动的分步注册向导-附源码
2015/08/26 Javascript
javascript中new关键字详解
2015/12/14 Javascript
jQuery制作图片旋转效果
2017/02/02 Javascript
浅析为什么a="abc" 不等于 a=new String("abc")
2017/10/25 Javascript
vue2 前端搜索实现示例
2018/02/26 Javascript
JS实现的图片选择顺序切换和循环切换功能示例【测试可用】
2018/12/28 Javascript
浅谈对于react-thunk中间件的简单理解
2019/05/01 Javascript
vue组件三大核心概念图文详解
2019/05/30 Javascript
js实现简单的贪吃蛇游戏
2020/04/23 Javascript
javascript实现贪吃蛇小游戏
2020/07/28 Javascript
nuxt.js添加环境变量,区分项目打包环境操作
2020/11/06 Javascript
[02:39]DOTA2国际邀请赛助威团西雅图第一天
2013/08/08 DOTA
python应用程序在windows下不出现cmd窗口的办法
2014/05/29 Python
Python实现爬取百度贴吧帖子所有楼层图片的爬虫示例
2018/04/26 Python
深入理解Django-Signals信号量
2019/02/19 Python
python代码 FTP备份交换机配置脚本实例解析
2019/08/01 Python
python logging模块的使用
2020/09/07 Python
HTML5全屏(Fullscreen)API详细介绍
2015/04/24 HTML / CSS
英国拖鞋购买网站:Bedroom Athletics
2020/02/28 全球购物
矿泉水广告词
2014/03/20 职场文书
社区关爱留守儿童活动方案
2014/08/22 职场文书
乡镇领导班子四风对照检查材料
2014/09/27 职场文书
优秀教师先进事迹材料
2014/12/15 职场文书
化妆品促销活动总结
2015/05/07 职场文书
SQLServer2019 数据库的基本使用之图形化界面操作的实现
2021/04/08 SQL Server
MySQL 外键约束和表关系相关总结
2021/06/20 MySQL