ECMAScript

新的数组api

findLast

反向迭代数组,并返回满足提供的测试函数的第一个元素的值,如果没有找到返回undefined

  1. const isEven = (number) => number % 2 === 0;
  2. const numbers = [1, 2, 3, 4];
  3. console.log(numbers.findLast(isEven)); // 4

findLastIndex

反向迭代数组,并返回满足所提供的测试函数的第一个元素的索引。若没有找到对应元素,则返回 -1

  1. const isEven = (number) => number % 2 === 0;
  2. const numbers = [1, 2, 3, 4];
  3. console.log(numbers.findLastIndex(isEven)); // 3

如果预计查找的元素靠后,使用findLastfindLastIndex 可以减少查询次数,提高程序效率。

toSorted

sort方法的复制版本,复制原数组,然后对新数组排序,并返回新数组。
排序逻辑和参数都和sort方法一样

  1. const array = [3, 2, 1];
  2. const sortedArray = array.toSorted();
  3. console.log(sortedArray); // [1, 2, 3]
  4. console.log(array); // 原数组不变 [3, 2, 1]

toReversed

reverse方法的复制版本;复制数组原数组,对新数组颠倒顺序,并返回新数组。

  1. const original = [1, 2, 3, 4];
  2. const reversed = original.toReversed();
  3. console.log(original); // [ 1, 2, 3, 4 ] 原数组不变
  4. console.log(reversed); // [ 4, 3, 2, 1 ]

toSpliced

splice() 方法的复制版本。它返回一个新数组,并在给定的索引处删除和/或替换了一些元素。

  1. const original = ["Jan", "Mar", "Apr", "May"];
  2. // 在索引 1 处添加一个元素
  3. const spliced = months.toSpliced(1, 0, "Feb");
  4. console.log(spliced); // ["Jan", "Feb", "Mar", "Apr", "May"]
  5. console.log(original); // ["Jan", "Mar", "Apr", "May"] 原数组不变

with

使用方括号表示法修改指定索引值的复制方法版本

  1. const arr = [1, 2, 3, 4, 5];
  2. console.log(arr.with(2, 6)); // [1, 2, 6, 4, 5]
  3. console.log(arr); // [1, 2, 3, 4, 5] 原数组不变

📌toSortedtoReversedtoSplicedwith在处理数组的时候会将稀疏数组的空值转成undefined

  1. const arr = [1, , 3, 4, , 6];
  2. console.log(arr.with(0, 2)); // [2, undefined, 3, 4, undefined, 6]

📌toSortedtoReversedtoSplicedwith可以使用call或者apply在类数组对象上应用

  1. function fn1(){
  2. console.log(Array.prototype.with.call(arguments,2,6)); // [1,2,6,4,5]
  3. }
  4. fn1(1,2,3,4,5)
  5. const arrayLike = {
  6. length: 3,
  7. unrelated: "foo",
  8. 0: 5,
  9. 2: 4,
  10. };
  11. console.log(Array.prototype.with.call(arrayLike, 0, 1));
  12. // [ 1, undefined, 4 ]

**toSorted****toReversed****toSpliced****with**在实际工作中是非常有用的,尤其在react中使用immutable data进行状态更新的时候,这四个api能很大程度简化操作,下面的例子是一个简单的todo组件例子。

  1. function App(){
  2. const [items,setItems] = React.useState([{id:1,text:'学习',done:false},{id:2,text:'吃饭',done:false},{id:3,text:'睡觉',done:false}]);
  3. const removeItem = (index)=>{
  4. // 方法一 使用splice
  5. // setItems((prev)=>{
  6. // const copy = [...prev];
  7. // copy.splice(index,1);
  8. // return copy;
  9. // })
  10. // 方法二 使用filter
  11. // setItems(prev=>prev.filter((_,i)=>i !== index ))
  12. // 方法三 使用toSpliced
  13. setItems(prev=>prev.toSpliced(index,1));
  14. }
  15. const toggleStatus = (index)=>{
  16. // 方法一 使用map
  17. // setItems(prev=>prev.map((item,i)=>{
  18. // if(index === i){
  19. // return {
  20. // ...item,
  21. // done: !item.done
  22. // }
  23. // }
  24. // return item;
  25. // }))
  26. // 使用with
  27. setItems(prev=>prev.with(index, {
  28. ...prev[index],
  29. done: !prev[index].done
  30. }))
  31. }
  32. return (
  33. <div>
  34. {
  35. items.map((i,index)=>{
  36. return (
  37. <div key={i.id}>
  38. <input type="checkbox" checked={i.done} onChange={()=>toggleStatus(index)}/>
  39. <span>{i.text}</span>
  40. <button onClick={()=>removeItem(index)}>删除</button>
  41. </div>
  42. )
  43. })
  44. }
  45. </div>
  46. )
  47. }

WeakMap 支持 Symbol 作为键

  1. const weak = new WeakMap();
  2. const key = Symbol("ref");
  3. weak.set(key, "ECMAScript 2023");
  4. console.log(weak.get(key)); // ECMAScript 2023

Hashbang

这个在nodejs中已经早就应用了,其实规范落地没什么影响。

  1. #!/usr/bin/env node
  2. console.log('hi');

兼容性

开发环境支持:
目前最新版的ts支持findLast和findLastIndex, 但是还不支持toSplicedtoSortedtoReversedwith,在typescript仓库中已经有人提了pull request;如果等不及官方支持,可在项目中给global的Array添加对应的类型定义,参考pr:https://github.com/microsoft/TypeScript/pull/51367/files
2023款ECMAScript - 图1
常用编辑器vscode和webstorm中语法还未支持,没有相关的语法提示,并且会有警告提示。
浏览器支持:
2023款ECMAScript - 图2
polyfill
目前core-js已经支持数组的api,但尚不支持WeakMap使用Symbol作为key来使用。
参考:https://github.com/tc39/proposals/blob/main/finished-proposals.md

参考资料

add change array by copy types: https://github.com/microsoft/TypeScript/pull/51367/files
TC39 Finished Proposals: https://github.com/tc39/proposals/blob/main/finished-proposals.md