面试官问:了解哪些最新的 ES 新特性?——这样回答更好!
目标
- 帮助大家答好这个问题,答出能给自己加分的答案。
- 真正帮助大家在实际工作中运用新特性,收获成长。
为什么会有这个问题?
首先要明白,面试官为什么会问这个问题?
这背后其实是希望了解:
- 你是否关注技术前沿,跟随行业趋势。
- 你是否能够将新特性运用到实际工作中,为团队和项目带来价值。
具体来说,了解和使用最新的 ECMAScript(ES)特性对前端开发者有以下几个显著好处:
1. 提高开发效率
- 语法更简洁:如
箭头函数
、模板字符串
、解构赋值
等,大幅减少代码量,提升可读性和可维护性。 - 功能更强大:例如可选链(
?.
)和空值合并操作符(??
),简化了对深层对象属性的安全访问,减少冗余代码。
2. 提升代码性能
- 新特性通常经过引擎优化,可以更高效地运行。
- 异步操作:
Async/Await
替代回调函数,避免“回调地狱”,更自然地管理异步操作。 - 迭代器与生成器:流式处理大规模数据,节省内存。
- 异步操作:
3. 解决复杂场景
- 大规模数据处理:
Map
、Set
和WeakMap
等数据结构在处理复杂数据时效率更高。 - 高级编程模式:如类的私有字段和方法(
#field
)、动态导入(import()
)、模块化等特性,提升了代码封装性。
4. 更好地支持现代浏览器
- 最新浏览器基本上支持 ES 的大多数新特性,无需额外的 polyfill 或转译,提升了代码的兼容性。
- 比如,使用模块化(
import/export
)可以直接加载模块,无需复杂的脚本管理工具。
5. 提高团队协作效率
- 使用最新语言特性可以减少代码风格分歧,让团队更容易理解代码意图。
- 默认参数减少无意义的条件判断。
Promise.allSettled
高效处理多个异步任务的结果。
6. 保持行业竞争力
- 前端技术迭代迅速,熟悉最新 ES 特性能够帮助开发者跟上技术趋势,增强行业竞争力。
- 现代开发框架(如 React、Vue、Svelte)和工具链(如 Webpack、Vite、Babel、ESLint)也高度依赖这些特性。
了解这些,希望可以真正的在实际应用中有所积淀,才能在面试回答中,游刃有余。
答题思路
在回答面试官关于“了解哪些最新的 ES 新特性”这一问题时,以下几个策略可以帮助你展现专业性和实际应用能力:
1. 结构清晰,突出重点
先概括性地回答“对 ECMAScript 的更新较为熟悉,特别是从 ES2021 到最近 ES2024 的新特性,并在项目中有实际应用。”
然后按照时间倒序,挑选出一些你熟悉且常用的特性进行介绍。(突显问题的最新字眼)
2. 挑选常见且有实际应用价值的特性
下文整理了一些时间倒序排列的 ES 新特性,可以根据你的熟悉程度和实际使用经验选择性回答。可能面试官会拓展,选择某一个的时候,尽量深入一点,自身也尽量围绕可能的扩展来回答。
3. 强调实际项目中的应用
不仅仅是列举特性,而是结合项目经验说明这些特性如何帮助解决实际问题。
以下列举的例子做参考,自己也可以依据自身情况,得出适合自己的回答:
- “在处理异步接口调用时,我主要使用了 ES2017 的
async/await
,极大地优化了回调地狱的问题。” - “ES2021 的逻辑赋值运算符在状态管理中非常有用,比如条件赋值 Redux store 的默认值。”
- ......
4. 展示学习态度和广度
如果时间允许,可以提到你学习新特性的方式,比如:
- 关注 TC39 提案进展,了解即将标准化的功能。
- 经常使用 Babel、Vite 等工具测试提案阶段的特性。
5. 避免“踩坑”
- 如果不熟悉某个特性,坦诚承认:“对这个特性了解不多,但我愿意快速学习并尝试使用。”
- 不要只背诵特性,而是注重描述实际应用场景。
示例回答
我对 ECMAScript 的更新一直有关注,并在项目中应用了许多新特性。
例如:
- ES2021 的逻辑赋值运算符(
&&=
,||=
,??=
)简化了 Redux 的状态更新逻辑; - ES2020 的可选链和空值合并操作符则优化了嵌套属性的访问,减少了手写的空值判断代码。
- 此外,我也在研究 ES2024 的 JSON 模块特性,可以直接
import
JSON 文件到代码中,这对管理配置文件非常实用。 - ......
这样的回答不仅展现了对语言特性的了解,还体现了实际开发经验和学习能力。
ES 特性整理
ES2024 (ES15)
Symbol.prototype.description
新增对符号的描述字段的访问支持,可以更轻松获取 Symbol 的描述信息。Array.prototype.toSorted
提供非破坏性排序方法,返回排序后的新数组而不改变原数组。RegExp V flag
引入“v”标志来支持 Unicode 正则表达式模式匹配和更复杂的匹配。Iterator Helpers
为迭代器对象添加常用工具方法,如map
、filter
、take
等,提高对流式数据处理的支持。JSON 模块(Import JSON)
可以直接通过import
导入 JSON 文件,无需额外配置工具。
示例:import data from './data.json' assert { type: 'json' }; console.log(data);
应用:用在配置文件或静态数据加载中,避免手动解析 JSON。
Object.groupBy()
用于根据回调函数返回的字符串值对可迭代对象(如数组)的对象元素进行分组。它返回一个对象,其中每个组名称作为键,相应的元素数组作为值。返回的对象和原始对象中的元素是相同的。即,如果更改元素的内部结构,它将反映在原始对象和返回的对象中。
const persons = [ {name:"John", age:70}, {name:"Kane", age:5}, {name:"Jack", age:50}, {name:"Rambo", age:15} ]; function callbackFunc({ age }) { if(age >= 60) { return "senior"; } else if(age > 17 && age < 60) { return "adult"; } else { return "kid"; } } const result = Object.groupBy(persons, callbackFunc); console.log("Kids: "); for (let [x,y] of result.kid.entries()) { console.log(y.name + " " + y.age); } const result2 = Map.groupBy(persons, callbackFunc); console.log("Kids2: "); for (let x of result2.get("kid")) { console.log(x.name + " " + x.age); }
Temporal API
是一种用于处理日期和时间的现代 API,用于取代原始的 Date API。它提供了一种更全面且用户友好的方式来处理日期和时间操作。
Temporal.PlainDate Temporal.PlainTime Temporal.PlainDateTime Temporal.PlainYearMonth Temporal.PlainMonthDay Temporal.ZonedDateTime
Promise withResolvers
是一个静态方法工厂,返回一个包含新 Promise 的对象以及两个函数,一个用于解析,另一个用于拒绝。这两个函数对应于初始代码片段中所示的传递给 Promise() 构造函数的执行器的两个参数。
const { promise, resolve, reject} = Promise.withResolvers(); setTimeout(() => { Math.random() > 0.5 ? resolve("Success") : reject("Error")},1000); promise.then(result => console.log(result)).catch(error => console.error(error));
ES2023 (ES14)
Array.prototype.findLast & Array.prototype.findLastIndex
两个的数组新方法,用于从最后一个元素搜索数组元素。它们的功能与find() 和findIndex() 类似,但搜索从数组末尾开始。这些方法可在Array和TypedArray原型上使用。此功能通过消除手动数组反转的过程,为逆序搜索提供了一种有效的方法。const isOdd = (number) => number % 2 === 1; const numbers = [1, 2, 3, 4, 5]; console.log(numbers.findLast(isOdd)); // 5 console.log(numbers.findLastIndex(isOdd)); // 4
Hashbang 支持
在脚本开头使用#!
来支持直接在脚本中定义解释器。 (也称为 shebang)语法已支持在可执行脚本的开头使用一系列字符(#!)
来定义要运行的程序的解释器。换句话说,此语法有助于告诉操作系统在执行脚本时使用哪个解释器。#!/usr/bin/env node 'use strict'; console.log("Hello world from hashbang syntax");
Symbol.prototype.isWellKnownSymbol
判断 Symbol 是否为“知名符号”。Change Array by Copy
非破坏性数组方法。toReversed() 、 toSorted() 、 toSpliced和with() 方法,这些方法返回新的数组副本而不是改变原始数组。const numbers = [1, 3, 2, 4, 5]; // toReversed const reversedArray = numbers.toReversed(); console.log(reversedArray); // [5, 4, 2, 3, 1] console.log(numbers); // [1, 3, 2, 4, 5] // toSorted const sortedArray = numbers.toSorted(); console.log(sortedArray); // [1, 2, 3, 4, 5] console.log(numbers); // [1, 3, 2, 4, 5] // toSpliced const splicedArray = numbers.toSpliced(1, 3); console.log(splicedArray); // [1, 5] console.log(numbers); // [1, 3, 2, 4, 5] // with const replaceWithArray = numbers.with(2, 10); console.log(replaceWithArray); // [1, 3, 10, 4, 5] console.log(numbers); // [1, 3, 2, 4, 5]
Symbols 作为 weakmap keys
在 ES2023 之前,
WeakMap
仅限于允许对象作为键,因为对象是唯一的且无法重新创建。由于Symbols
是 ECMAScript 中唯一允许唯一值的基元,因此 WeakMap API 已使用符号作为键进行扩展,而不仅仅是使用对象。const weak = new WeakMap(); const objKey = { x:10 }; weak.set(objKey, "ES2023"); console.log(weak.get(objKey)); //ES2023 const key = Symbol("ref"); weak.set(key, "ES2023"); console.log(weak.get(key)); //ES2023
ES2022 (ES13)
Top-Level Await
在模块的顶层支持使用await
,简化异步操作。import posts from './posts'; const getPosts = async() => { let posts = await posts(); return posts; } let posts = await posts();
动态依赖路径: 当您拥有依赖于运行时值的依赖项的动态路径时,await 有助于在运行时加载或导入消息。
const messages = await import(`./messages-${language}.js`);
依赖回退: 如果导入的模块加载失败,则加载后备模块用于加载依赖项。
let lodash; try { lodash = await import('https://first.domain.com/lodash'); } catch { lodash = await import('https://second.domain.com/lodash'); }
资源初始化: 此功能可用于使用数据库初始化应用程序。
import { dbConnector} from './dbUtils.js' //connect to database const connection = await dbConnector.connect(); export default function(){ connection.list() }
RegExp Match Indices
在正则匹配结果中返回每个匹配的索引范围。附加信息包括 RegExp 中匹配的开始和结束索引以及在输入字符串中使用\d
标志。const regexPatter = /Jack/g; const input = 'Authos: Jack, Alexander and Jacky'; const result = [...input.matchAll(regexPatter)]; console.log(result[0]); // ['Jack', index: 8, input: 'Authos: Jack, Alex and Jacky', groups: undefined] // \d const regexPatter = /(Jack)/gd; const input = 'Authos: Jack, Alexander and Jacky'; const result = [...input.matchAll(regexPatter)]; console.log(result[0]); // ['Jack', 'Jack', index: 8, input: 'Authos: Jack, Alexander and Jacky', groups: undefined, indices: Array(2)]
class
公共实例字段 & 私有字段(#field
)静态字段(static)
实现类的更强封装性,使用私有字段保护数据。
示例:class Person { #privateField = 'secret'; publicField = 'visible'; static #employerName="Github" static #getEmployerName() { return #employerName } }
应用:在模块化开发中保护内部状态。
Array .at()
.at()
方法用于通过传递负索引值来访问数组或字符串元素。即,它允许从数组末尾或字符串访问值。const array = [1, 2, 3, 4, 5]; console.log(array.at(-2)); // 4 const string = '12345'; console.log(string.at(-2));
Error Cause
cause
属性作为额外参数添加到 Error() 构造函数中,允许将错误链接起来,类似于错误链中类似 Java 的堆栈跟踪。function processUserData(arrayData) { return arrayData.map(data => { try { const json = JSON.parse(data); return json; } catch (err) { throw new Error( `Data processing failed`, {cause: err} ); } }); }
hasOwn
新的
Object.hasOwn()
方法是Object.prototype.hasOwnProperty
的替换或改进版本。它是一个静态方法,如果指定对象将指定的属性作为其自己的属性,则返回 true。如果该属性是继承的或不存在,则该方法返回 false。当
hasOwnProperty
被覆盖时:在某些情况下,您需要在对象上定义自定义的
hasOwnProperty
。当你尝试应用hasOwnProperty
来确定是否拥有自己的属性时,它会抛出错误,如下例所示。const user = { age: 35, hasOwnProperty: ()=> { return false; } }; user.hasOwnProperty('age') // throws a TypeError user.hasOwn('age') // true
使用 create(null) 函数创建一个对象:
如果您借助 create(null) 函数创建新对象,则新创建的对象不会继承自 Object.prototype。所以它没有 hasOwnProperty 方法。
const user = Object.create(null); user.age = 35; user.hasOwnProperty('age'); // throws a TypeError user.hasOwn('age'); // true
ES2021 (ES12)
String.prototype.replaceAll
提供全局字符串替换功能,避免手动正则表达式。用于将某个字符串的所有出现位置替换为另一个字符串值。早些时候,如果不使用正则表达式,就不可能替换子字符串的所有实例。
console.log('10101010'.replace(new RegExp('0', 'g'), '1')); // 11111111 console.log('01010101'.replace(/0/g, '1')); // 11111111 // replaceAll console.log('10101010'.replaceAll('0', '1')); // 11111111 console.log('01010101'.replaceAll('0', '1')); // 11111111
Promise.any
返回第一个完成的 Promise(无论是否成功),更适合处理多个异步任务。let promise1 = new Promise((resolve) => setTimeout(resolve, 100, 'Resolves after 100ms')); let promise2 = new Promise((resolve) => setTimeout(resolve, 200, 'Resolves after 200ms')); let promise3 = new Promise((resolve, reject) => setTimeout(reject, 0) ); let promises = [promise1, promise2, promise3]; Promise.any(promises) .then( value => console.log(value)); // Resolves after 100ms
如果没有任何承诺得到解决,那么它会抛出
AggregateError
异常。(async () => { try { const output = await Promise.any([ Promise.reject('Error 1'), Promise.reject('Error 2'), Promise.reject('Error 3'), ]); console.log(`Output: ${output}`); } catch (err) { console.log(`Error: ${err.errors}`); } })(); // Error: Error1,Error2,Error3
逻辑赋值运算符
引入&&=
、||=
和??=
运算符,简化常见逻辑操作。WeakRefs
引入弱引用和垃圾回收通知机制,提升对资源管理的控制。WeakRef 提供了两项新功能
- 使用 WeakRef 类创建对对象的弱引用
- 在对象被垃圾收集后,使用 FinalizationRegistry 类运行用户定义的终结器
WeakRef 对一个对象的引用,如果它是内存中该对象的唯一引用,则不会阻止垃圾回收。当我们不想将对象永远保留在内存中时(例如,WebSocket),它很有用。弱引用的主要用途是实现到大型对象的缓存或映射,对于很少使用的对象,您不需要将其保留在内存中。
在 ES12 之前,WeakMap 和 WeakSet 是 JavaScript 中弱引用对象的唯一方法。而 ES12 中的 WeakRef 提供了实际的弱引用,从而为了解对象的生命周期提供了一个窗口。
const myObject = new WeakRef({ name: ‘Sudheer’, age: 34 }); console.log(myObject.deref()); //output: {name: “Sudheer”, age: 35} console.log(myObject.deref().name); //output: Sudheer
FinalizationRegistry
允许您在对象被垃圾收集时请求回调。它用作清理回调。
// Create new FinalizationRegistry: const reg = new FinalizationRegistry((val) => { console.log(val); }); (() => { // Create new object: const obj = {} // Register finalizer for the "obj" as first argument and value for callback function as second argument: reg.register(obj, 'obj has been garbage-collected.') })();
注意: 完成回调不会在垃圾收集事件侦听器后立即运行,因此不要将其用于重要的逻辑或指标。
逻辑赋值运算符(
&&=
,||=
,??=
)
逻辑赋值运算符将逻辑运算(&&、|| 或 ??)与赋值相结合。它们对于为变量分配默认值非常有用。 示例:let x = 10; let y = 20; x &&= y; console.log(x); // 20 let name = ''; name ||= 'Default'; // 如果 name 是 false,赋值 'Default' let x; let y = 1; x ??= y; console.log(x); // 1
Numeric Separators
数字分隔符通过使用下划线 (_) 提供数字之间的分隔,有助于在 JavaScript 中读取大数字(或数字文字)。换句话说,通过在数字组之间创建视觉分隔,数字文字更具可读性。
// 使用 _ 数字分隔符使十亿和一万亿变得更易读 const billion = 1000_000_000; console.log(billion); // 1000000000 const trillion = 1000_000_000_000n; // BigInt number console.log(trillion); // 1000000000000 // 也可用于二进制和十六进制文字 const binaryLiteral = 0b1010_1010; console.log(binaryLiteral); const hexLiteral = 0xFF_FF_FF_FF; console.log(hexLiteral);
ES2020 (ES11)
BigInt
在早期的 JavaScript 版本中,使用 Number 类型存在限制。即,您无法安全地表示大于 pow(2, 53) 的整数值。
在 ES2020 中,
BigInt
被引入作为第七种基本类型来表示大于 pow(2, 53) - 1(或 9007199254740991 或 Number.MAX_SAFE_INTEGER)的整数(任意精度的整数)。这是通过将
n
附加到整数文字的末尾或通过调用函数 BigInt() 创建的。// 1. Current number system const max = Number.MAX_SAFE_INTEGER; console.log(max + 1) // 9007199254740992 console.log(max + 2) // 9007199254740992 // 2. BigInt representation const bigInt = 9007199254740991n; const bigIntConstructorRep = BigInt(9007199254740991); // 9007199254740991n const bigIntStringRep = BigInt("9007199254740991"); // 9007199254740991n // 3. Typeof usage console.log(typeof 1)// number console.log(typeof 1n)// bigint console.log(typeof BigInt('1'))// bigint // 4. Operators const previousMaxNum = BigInt(Number.MAX_SAFE_INTEGER); console.log(previousMaxNum + 2n); //9007199254740993n (this was not possible before) console.log(previousMaxNum -2n); //9007199254740990n console.log(previousMaxNum * 2n); //18014398509481982n console.log(previousMaxNum % 2n); //1n console.log(previousMaxNum / 2n); // 4503599627370495n // 5. comparison console.log(1n === 1); // false console.log(1n === BigInt(1)); // true console.log(1n == 1); // true
Dynamic Import
动态加载模块,通过import()
实现异步模块加载。dynamic import
以有条件或按需加载模块。由于它返回所请求模块的模块命名空间对象的承诺,因此现在可以使用 async/await 将模块解析或导入分配给变量,如下所示<script> const moduleSpecifier = './message.js'; import(moduleSpecifier) .then((module) => { module.default(); // Hello, default export module.sayGoodBye(); //Bye, named export }) .catch(err => console.log('loading error')); </script>
<script> (async function() { const moduleSpecifier = './message.js'; const messageModule = await import(moduleSpecifier); messageModule.default(); // Hello, default export messageModule.sayGoodBye(); //Bye, named export })(); </script>
导入的模块同时显示默认导出和命名导出
export default () => { return "Hello, default export"; } export const sayGoodBye = () => { return "Bye, named export" }
注意: 动态导入不需要
type="module"
的脚本Nullish Coalescing Operator (??)
提供更安全的空值判断操作符。空值合并运算符(
??
)是一个逻辑运算符,当左侧的操作数为null
或者undefined
时,返回其右侧操作数,否则返回左侧操作数。const foo = null ?? 'default string'; console.log(foo); // Expected output: "default string" const baz = 0 ?? 42; console.log(baz); // Expected output: 0
Optional Chaining (?.)
优化深层属性访问,避免手动检查每个层级。可选链运算符(
?.
) 用于访问对象的属性或调用函数。如果使用此运算符访问的对象或调用的函数是undefined
或null
,则表达式会短路并计算为undefined
,而不是抛出错误。const adventurer = { name: 'Alice', cat: { name: 'Dinah', }, }; const dogName = adventurer.dog?.name; console.log(dogName); // Expected output: undefined console.log(adventurer.someNonExistentMethod?.()); // Expected output: undefined
Promise.allSettled
等待所有 Promise 结果(成功或失败)完成后返回。静态方法将一个 Promise 可迭代对象作为输入,并返回一个单独的
Promise
。当所有输入的 Promise 都已敲定时(包括传入空的可迭代对象时),返回的 Promise 将被兑现,并带有描述每个 Promise 结果的对象数组。const promise1 = Promise.resolve(3); const promise2 = new Promise((resolve, reject) => setTimeout(reject, 100, 'foo'), ); const promises = [promise1, promise2]; Promise.allSettled(promises).then((results) => results.forEach((result) => console.log(result.status)), ); // Expected output: // "fulfilled" // "rejected"
可选链(
?.
)
访问嵌套对象属性时避免undefined
或null
错误。
示例:const user = { profile: { name: 'Alice' } }; console.log(user?.profile?.name);
String matchAll
matchAll()
方法返回一个迭代器,该迭代器包含了检索字符串与正则表达式进行匹配的所有结果(包括捕获组)。const regexp = /t(e)(st(\d?))/g; const str = 'test1test2'; const array = [...str.matchAll(regexp)]; console.log(array[0]); // Expected output: Array ["test1", "e", "st1", "1"] console.log(array[1]); // Expected output: Array ["test2", "e", "st2", "2"]
globalThis
在 ES2020 之前,仅仅为了访问全局对象就需要在不同的JavaScript环境(跨平台)中编写不同的语法。对于开发人员来说这确实是一个困难时期,因为你需要在浏览器端使用
window, self, or frames
,在 Nodejs 上使用global
,在 Web Workers 端self
。另一方面,
this
关键字可以在非严格模式的函数内部使用,但在严格模式下它会给出 undefined 。如果您将Function('return this')()
视为上述环境的解决方案,那么对于启用 CSP 的环境(其中 eval() 被禁用),它将失败。var getGlobal = function () { if (typeof self !== 'undefined') { return self; } if (typeof window !== 'undefined') { return window; } if (typeof global !== 'undefined') { return global; } throw new Error('unable to locate global object'); }; var globals = getGlobal(); if (typeof globals.setTimeout !== 'function') { console.log('no setTimeout in this environment or runtime'); } // globalThis if (typeof globalThis.setTimeout !== 'function') { console.log('no setTimeout in this environment or runtime'); }
全局属性
globalThis
包含全局的this
值,类似于全局对象(global object)。function canMakeHTTPRequest() { return typeof globalThis.XMLHttpRequest === 'function'; } console.log(canMakeHTTPRequest()); // Expected output (in a browser): true
i
mport.meta i
对象是由 ECMAScript 实现使用 null 原型创建的,用于获取有关 JavaScript 模块的上下文特定元数据。假设您正在尝试从脚本加载mport.meta my-module
,<script type="module" src="my-module.js"></script> console.log(import.meta); // { url: "file:///home/user/my-module.js" }
注意: 记住
import
并不是真正的对象,但i
是作为可扩展的对象提供的,并且其属性是可写的、可配置的和可枚举的。mport.meta for..in order
在 ES2020 之前,规范没有指定 (a in b) 的运行顺序。尽管大多数 javascript 引擎/浏览器按照定义的顺序循环访问对象的属性,但并非所有情况都是如此。这已在 ES2020 中正式标准化。
var object = { 'a': 2, 'b': 3, 'c': 4 } for(let key in object) { console.log(key); // a b c }
ES2019 (ES10)
Array.prototype.flat & flatMap
提供数组扁平化操作及映射后扁平化的功能。flat()
方法,将嵌套数组“展平”到顶层数组中。该方法的功能类似于Lodash的_.flattenDepth()
函数。此方法接受一个可选参数,该参数指定嵌套数组应展平的级别数,默认嵌套级别为 1。注意: 如果数组中有任何空槽,它们将被丢弃。
const numberArray = [[1, 2], [[3], 4], [5, 6]]; const charArray = ['a', , 'b', , , ['c', 'd'], 'e']; const flattenedArrOneLevel = numberArray.flat(1); const flattenedArrTwoLevel = numberArray.flat(2); const flattenedCharArrOneLevel = charArray.flat(1); console.log(flattenedArrOneLevel); // [1, 2, [3], 4, 5, 6] console.log(flattenedArrTwoLevel); // [1, 2, 3, 4, 5, 6] console.log(flattenedCharArrOneLevel); // ['a', 'b', 'c', 'd', 'e']
而flatMap() 方法将
map()
和flat()
合并为一个方法。它首先使用给定函数的返回值创建一个新数组,然后连接该数组的所有子数组元素。const numberArray1 = [[1], [2], [3], [4], [5]]; console.log(numberArray1.flatMap(value => [value * 10])); // [10, 20, 30, 40, 50]
Object.fromEntries
将键值对列表转换为对象的快捷方法。// Object to Array: const obj = {'a': '1', 'b': '2', 'c': '3' }; const arr = Object.entries(obj); console.log(arr); // [ ['a', '1'], ['b', '2'], ['c', '3'] ]
// Array to Object: const arr = [ ['a', '1'], ['b', '2'], ['c', '3'] ]; let obj = {} for (let [key, val] of arr) { obj[key] = val; } console.log(obj); const arr = [ ['a', '1'], ['b', '2'], ['c', '3'] ]; const obj = Object.fromEntries(arr); console.log(obj); // { a: "1", b: "2", c: "3" }
此方法使用的常见情况之一是使用 URL 的查询参数,
const paramsString = 'param1=foo¶m2=baz'; const searchParams = new URLSearchParams(paramsString); Object.fromEntries(searchParams); // => {param1: "foo", param2: "baz"}
String.trimStart & trimEnd
去除字符串两端多余空格的便捷方法。为了与padStart/padEnd保持一致,ES2019提供了标准函数
trimStart
和trimEnd
来修剪字符串开头和结尾的空格。然而,为了网络兼容性(避免任何损坏),trimLeft
和trimRight
将分别是trimStart
和trimEnd
的别名。//Prior ES2019 let messageOne = " Hello World!! "; console.log(messageOne.trimLeft()); //Hello World!! console.log(messageOne.trimRight()); // Hello World!! //With ES2019 let messageTwo = " Hello World!! "; console.log(messageTwo.trimStart()); //Hello World!! console.log(messageTwo.trimEnd()); // Hello World!!
Optional Catch Binding
catch
块中捕获变量为可选项,允许忽略错误对象。// With binding parameter(<ES9) try { ··· } catch (error) { ··· } // Without binding parameter(ES9) try { ··· } catch { ··· }
Symbol description
ES2019 引入了只读描述属性来检索包含符号描述的字符串。
console.log(Symbol('one').description); // one console.log(Symbol.for('one').description); // "one" console.log(Symbol('').description); // '' console.log(Symbol().description); // unefined console.log(Symbol.iterator.description); // "Symbol.iterator"
JSON Improvements
JSON Superset
在 ES2019 之前,ECMAScript 声称 JSON 是 JSON.parse 的子集,但事实并非如此。因为与 JSON 字符串不同,ECMAScript 字符串文字不能包含字符
U+2028
(行分隔符)和U+2029
(段落分隔符)。如果您仍然使用这些字符,则会出现语法错误。作为解决方法,您必须使用转义序列将它们放入字符串中。eval('"\u2028"'); // SyntaxError
而 JSON 字符串可以同时包含 U+2028 和 U+2029 而不会产生错误。
console.log(JSON.parse('"\u2028"')); // ''
ES2019 中取消了此限制。这简化了规范,无需针对 ECMAScript 字符串文字和 JSON 字符串文字制定单独的规则。
格式良好的 JSON.Stringify(): 在 ES2019 之前,如果输入中存在任何单独代理,则使用 JSON.stringify 方法返回未形成的 Unicode 字符串(格式错误的 Unicode 字符串)。
console.log(JSON.stringify("\uD800")); // '"�"'
而在 ES2019 中,JSON.stringify 输出单独代理的转义序列,使其输出有效的 Unicode 并可以用 UTF-8 表示。
console.log(JSON.stringify("\uD800")); // '"\ud800"'
Function.toString()
函数有一个名为
toString()
的实例方法,它返回一个字符串来表示函数代码。以前版本的 ECMAScript 删除了函数代码中的空格、换行和注释,但在 ES2020 中保留了原始源代码。function sayHello(message) { let msg = message; //Print message console.log(`Hello, ${msg}`); } console.log(sayHello.toString()); // function sayHello(message) { // let msg = message; // //Print message // console.log(`Hello, ${msg}`); // }
ES2018 (ES9)
- Asynchronous Iteration
支持异步迭代协议,用于处理异步数据流。 - Promise.prototype.finally
为 Promise 增加finally
方法,便于清理操作。let isLoading = true; fetch('http://somesite.com/users') .then(data => data.json()) .catch(err => console.error(err)) .finally(() => { isLoading = false; console.log('Finished loading!!'); })
- Rest/Spread Properties
支持对象的解构与扩展操作符。function myfunc1({ a, ...x }) { console.log(a, x); // 1, { b: 2, c: 3, d:4 } } myfunc1({ a: 1, b: 2, c: 3, d: 4 }); const myObject = { a: 1, b: 2, c: 3, d:4 }; const myNewObject = { ...myObject, e: 5 }; // { a: 1, b: 2, c: 3, d: 4, e: 5 }
- RegExp 的增强
- 新增
s
修饰符(dotAll 模式)。 - 支持命名捕获组和反向引用。
- 支持后行断言。
- 新增
ES2017 (ES8)
SharedArrayBuffer 和 Atomics
增强对共享内存和多线程操作的支持。Atomics 是一个全局对象,它提供作为静态方法执行的原子操作。它们与 SharedArrayBuffer(固定长度二进制数据缓冲区)对象一起使用。这些方法的主要用例是,
原子操作: 当内存共享时,多个线程可以在内存中读写相同的数据。因此存在数据丢失的可能性。但原子操作可确保写入和读取可预测的值、操作在下一个操作开始之前完成并且操作不会中断。
它提供了静态方法,例如 add、or、and、xor、load、store、isLockFree 等
const sharedMemory = new SharedArrayBuffer(1024); const sharedArray = new Uint8Array(sharedMemory); sharedArray[0] = 10; Atomics.add(sharedArray, 0, 20); console.log(Atomics.load(sharedArray, 0)); // 30 Atomics.sub(sharedArray, 0, 10); console.log(Atomics.load(sharedArray, 0)); // 20 Atomics.and(sharedArray, 0, 5); console.log(Atomics.load(sharedArray, 0)); // 4 Atomics.or(sharedArray, 0, 1); console.log(Atomics.load(sharedArray, 0)); // 5 Atomics.xor(sharedArray, 0, 1); console.log(Atomics.load(sharedArray, 0)); // 4 Atomics.store(sharedArray, 0, 10); // 10 Atomics.compareExchange(sharedArray, 0, 5, 10); console.log(Atomics.load(sharedArray, 0)); // 10 Atomics.exchange(sharedArray, 0, 10); console.log(Atomics.load(sharedArray, 0)); //10 Atomics.isLockFree(1); // true
等待通知:
wait()
和notify()
方法都提供了等待直到某个条件变为真的方法,并且通常用作阻塞结构。// 定义共享内存和数组 const sharedMemory = new SharedArrayBuffer(1024); const sharedArray = new Int32Array(sharedMemory); // 读取线程正在睡眠并等待位置 0,该位置预计为 10。 // 在该值被写入线程覆盖后,您可以观察到不同的值。 Atomics.wait(sharedArray, 0, 10); console.log(sharedArray[0]); // 100 // 现在写入线程存储一个新值(例如,100)并通知等待线程, Atomics.store(sharedArray, 0, 100); Atomics.notify(sharedArray, 0, 1);
String.prototype.padStart & padEnd
增加字符串补全功能。padStart(): 使用此方法,填充应用于字符串的左侧或开头。
// 例如,出于安全原因,您可能只想显示信用卡号的最后四位数字, const cardNumber = '01234567891234'; const lastFourDigits = cardNumber.slice(-4); const maskedCardNumber = lastFourDigits.padStart(cardNumber.length, '*'); console.log(maskedCardNumber); // expected output: "**********1234"
padEnd(): 使用此方法,填充应用于字符串的右侧或结尾侧。
const label1 = "Name"; const label2 = "Phone Number"; const value1 = "John" const value2 = "(222)-333-3456"; console.log((label1 + ': ').padEnd(20, ' ') + value1); // Name: John console.log(label2 + ": " + value2); // Phone Number: (222)-333-3456
Async/Await
使异步操作写法更像同步代码,提升代码可读性。
示例:async function fetchData() { const response = await fetch('/api'); return response.json(); }
Object values
const countries = { IN: 'India', SG: 'Singapore', } Object.values(countries) // ['India', 'Singapore'] console.log(Object.values(['India', 'Singapore'])); // ['India', 'Singapore'] console.log(Object.values('India')); // ['I', 'n', 'd', 'i', 'a']
Object entries
const countries = { IN: 'India', SG: 'Singapore', } Object.entries(countries) // [["IN", "India"], ["SG", "Singapore"]] const countriesArr = ['India', 'Singapore']; console.log(Object.entries(countriesArr)); // [ ['0', 'India'], ['1', 'Singapore']] const country = 'India'; console.log(Object.entries(country)); // [["0", "I"], ["1", "n"], ["2", "d"], ["3", "i"], ["4", "a"]] console.log(Object.entries(100)); // [], an empty array for any primitive type because it won't have any own properties
Object property descriptors
Object.getOwnPropertyDescriptors()
方法返回给定对象的所有自己的属性描述符。- value: 与属性关联的值(仅限数据描述符)。
- writable: true 当且仅当与属性关联的值可以更改时
- get: 充当属性的 getter 的函数。
- set: 充当属性设置器的函数。
- configurable: 当且仅当该属性描述符的类型可以更改或删除时为 true。
- enumerable: 当且仅当该属性在属性枚举期间出现时才为 true。
const profile = { age: 42 }; const descriptors = Object.getOwnPropertyDescriptors(profile); console.log(descriptors); // {age: {configurable: true, enumerable: true, writable: true }}
尾随逗号
// 参数定义和函数调用中允许使用尾随逗号 function func(a,b,) { // declaration console.log(a, b); } func(1,2,); // invocation // 但如果函数参数定义或函数调用只包含逗号,则会抛出语法错误 function func1(,) { // SyntaxError: missing formal parameter console.log('no args'); }; func1(,); // SyntaxError: expected expression, got ','
注意: Rest 参数和 JSON 中不允许使用尾随逗号。
ES2016 (ES7)
Array.prototype.includes
数组中支持更直观的元素存在性检查。const array = [1,2,3,4,5,6]; if(array.includes(5)){ console.log("Found an element"); }
Array.prototype.includes()
比Array.prototype.indexOf()
方法更好地处理 NaN 和 Undefined 值。即,如果数组包含 NaN 和 Undefined 值,则在搜索 NaN 和 Undefined 时,indexOf()
不会返回正确的索引。let numbers = [1, 2, 3, 4, NaN, ,]; console.log(numbers.indexOf(NaN)); // -1 console.log(numbers.indexOf(undefined)); // -1
另一方面,
includes
方法能够找到这些元素let numbers = [1, 2, 3, 4, NaN, ,]; console.log(numbers.includes(NaN)); // true console.log(numbers.includes(undefined)); // true
指数操作符( )**
使用**
代替Math.pow
,提高数学操作的可读性。//Prior ES7 const cube = x => Math.pow(x, 3); console.log(cube(3)); // 27 //Using ES7 const cube1 = x => x ** 3; console.log(cube1(3)); // 27
ES2015 (ES6)
Let & Const
新增块级作用域变量声明方式。Arrow Functions
引入箭头函数,简化函数表达式语法。Template Literals
提供模板字符串Classes
同时可以使用
extend
关键字来使用继承对象解构
扩展运算符
Modules
Set
Set 是一个内置对象,用于存储任何类型的唯一值的集合。
Weakset
Map
Weakmap
Symbols
Proxies
Promises
Reflect
Array.find()和Array.findIndex()