JavaScript Array.from 详解

0x0 前言

Array.from:允许在 JavaScript 集合(如: 数组、类数组对象、或者是字符串、mapset 等可迭代对象) 上进行有用的转换。

Array.from(arrayLike[, mapFunction[, thisArg]])
  • arrayLike:必传参数,想要转换成数组的伪数组对象或可迭代对象。
  • mapFunction:可选参数,mapFunction(item,index){...} 是在集合中的每个项目上调用的函数。返回的值将插入到新集合中。
  • thisArg:可选参数,执行回调函数 mapFunctionthis 对象。这个参数很少使用。

0x1 将类数组转换成数组

Array.from() 第一个用途:将类数组对象转换成数组。平时会碰到的类数组对象有:函数中的 arguments 关键字,或者是一个 DOM 集合。

在下面的示例中,对函数的参数求和:

function sumArguments() {
    return Array.from(arguments).reduce((sum, num) => sum + num);
}
 
sumArguments(1, 2, 3); // => 6

Array.from(arguments) 将类数组对象 arguments 转换成一个数组,然后使用数组的 reduce 方法求和。

此外,Array.from() 的第一个参数可以是任意一个可迭代对象:

Array.from('Hey');                   // => ['H', 'e', 'y']
Array.from(new Set(['one', 'two'])); // => ['one', 'two']
 
const map = new Map();
map.set('one', 1)
map.set('two', 2);
Array.from(map); // => [['one', 1], ['two', 2]]

0x2 克隆数组

JavaScript 中有很多克隆数组的方法。Array.from() 可以很容易的实现数组的浅拷贝。

const numbers = [3, 6, 9];
const numbersCopy = Array.from(numbers);
 
numbers === numbersCopy; // => false

Array.from(numbers) 创建了对 numbers 数组的浅拷贝,numbers === numbersCopy 的结果是 false,意味着虽然 numbersnumbersCopy 有着相同的项,但是它们是不同的数组对象。

是否可以使用 Array.from() 创建数组的克隆,包括所有嵌套的?挑战一下!

function recursiveClone(val) {
    return Array.isArray(val) ? Array.from(val, recursiveClone) : val;
}
 
const numbers = [[0, 1, 2], ['one', 'two', 'three']];
const numbersClone = recursiveClone(numbers);
 
numbersClone; // => [[0, 1, 2], ['one', 'two', 'three']]
numbers[0] === numbersClone[0] // => false

recursiveClone() 能够对数组的深拷贝,通过判断 数组的 item 是否是一个数组,如果是数组,就继续调用 recursiveClone() 来实现了对数组的深拷贝。

你能编写一个比使用 Array.from() 递归拷贝更简短的数组深拷贝吗?如果可以的话,请写在下面的评论区。

0x3 使用值填充数组

如果需要使用相同的值来初始化数组,那么 Array.from() 将是不错的选择。定义一个函数,创建一个填充相同默认值的数组:

const length = 3;
const init   = 0;
const result = Array.from({ length }, () => init);
 
result; // => [0, 0, 0]

result 是一个新的数组,它的长度为3,数组的每一项都是0。调用 Array.from() 方法,传入一个类数组对象 { length } 和 返回初始化值的 mapFunction 函数。

但是,有一个替代方法 array.fill() 可以实现同样的功能。

const length = 3;
const init   = 0;
const result = Array(length).fill(init);
 
fillArray2(0, 3); // => [0, 0, 0]

fill() 使用初始值正确填充数组。

0x4 使用对象填充数组

当初始化数组的每个项都应该是一个新对象时,Array.from() 是一个更好的解决方案:

const length = 3;
const resultA = Array.from({ length }, () => ({}));
const resultB = Array(length).fill({});
 
resultA; // => [{}, {}, {}]
resultB; // => [{}, {}, {}]
 
resultA[0] === resultA[1]; // => false
resultB[0] === resultB[1]; // => true

Array.from 返回的 resultA 使用不同空对象实例进行初始化。之所以发生这种情况是因为每次调用时,mapFunction,即此处的 () => ({}) 都会返回一个新的对象。

然后,fill() 方法创建的 resultB 使用相同的空对象实例进行初始化。不会跳过空项。

0x5 生成数字范围

可以使用 Array.from() 生成值范围。例如,下面的 range 函数生成一个数组,从0开始到 end - 1

function range(end) {
    return Array.from({ length: end }, (_, index) => index);
}
 
range(4); // => [0, 1, 2, 3]

range() 函数中,Array.from() 提供了类似数组的 {length:end} ,以及一个简单地返回当前索引的 map 函数 。这样就可以生成值范围。

0x6 数组去重

由于 Array.from() 的入参是可迭代对象,因此可以利用其与 Set 结合来实现快速从数组中删除重复项。

function unique(array) {
  return Array.from(new Set(array));
}
 
unique([1, 1, 2, 3, 3]); // => [1, 2, 3]

首先,new Set(array) 创建了一个包含数组的集合,Set 集合会删除重复项。

因为 Set 集合是可迭代的,所以可以使用 Array.from() 将其转换为一个新的数组。

这样就实现了数组去重。

MDN Web Docs

  • 11
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Array.from和Array.of是两个不同的Array构造函数的静态方法。 1. Array.from: - Array.from方法用于从可迭代对象或类数组对象创建一个新的数组实例。 - 它接受一个类数组对象或可迭代对象作为第一个参数,可以是类似数组的对象(拥有length属性和索引)或可迭代对象(如Set、Map、字符串等)。 - 可选的第二个参数是一个映射函数,用于对每个元素进行映射转换。 - 可选的第三个参数是指定映射函数中this的值。 - 返回一个新的数组实例。 示例使用Array.from方法创建一个新的数组实例: ```javascript const str = 'hello'; const arr = Array.from(str); console.log(arr); // ['h', 'e', 'l', 'l', 'o'] ``` 2. Array.of: - Array.of方法用于创建一个新的数组实例,不论传递的参数数量是多少。 - 它接受任意数量的参数作为数组的元素值,并返回一个包含这些元素的新数组实例。 - 与Array构造函数不同,如果只传递一个参数且该参数为数字类型,则会创建一个具有指定长度的空数组,而不是包含该数字作为唯一元素值的数组。 示例使用Array.of方法创建新的数组实例: ```javascript const arr1 = Array.of(1, 2, 3); console.log(arr1); // [1, 2, 3] const arr2 = Array.of(5); console.log(arr2); // [5] const arr3 = Array.of(5, 'hello', true); console.log(arr3); // [5, 'hello', true] ``` 总结: - Array.from方法用于从可迭代对象或类数组对象创建新的数组实例,并可选择进行映射转换。 - Array.of方法用于创建一个新的数组实例,不论传递的参数数量是多少。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值