-
Notifications
You must be signed in to change notification settings - Fork 1.4k
return largest numbers in arrays
S1ngS1ng edited this page Mar 19, 2017
·
1 revision
- 这个
function
接收多个数组为参数,返回每个数组的最大值组成的新数组 - 这道题不会涉及到特殊情况的处理,因此只需要遍历,找出最大值就可以了
- 对于初级解法,没什么太多要参考的。只需要想想,如何从数组中提取出最大值就可以了
- 题目的整体思路与之前的 找出最长单词(Find the Longest word in a String) 十分相似。区别在于,这道题是在操作二维数组,那道题是在操作一维数组
- 首先设置一个空数组,用于存储结果
- 遍历一遍传入的参数,参数是一个二维数组。因此,在遍历的过程中,我们操作的对象就是其中的每一个数组
- 然后,提取出当前数组的最大值,并把它添加到第一步初始化的数组中。最后返回这个数组即可
- 获取数组的最大值也很容易,首先在外面设置一个变量,初始化为 0。然后,遍历一遍数组,发现更大的就赋值给它,没发现更大的就什么也不做。遍历结束后,这个变量就是数组最大值
- 基本思路中,我们先不涉及太多的技巧,就用最直观、最简单的方式,先把它写出来
function largestOfFour(arr) {
// 空数组,用于存储结果
var result = [];
// 遍历传入的数组
for (var i = 0; i < arr.length; i++) {
// 初始化最大值,可以想一下为什么要写在这里
var largest = 0;
// arr 是一个二维数组,arr[i] 才是我们要找最大值的每一个数组
for (var j = 0; j < arr[i].length; j++) {
if (arr[i][j] > largest) {
// 找到更大的值,更新 largest
largest = arr[i][j];
}
}
// 把最大值存储到 `result`
result.push(largest);
}
return result;
}
- 首先要想明白一点,第一层的
for
循环我们是在遍历传入的参数。第二层的才是每一个数组。比如arr[0]
就是传入的二维数组中的第一个数组,那么arr[0][0]
就是传入的二维数组中,第一个数组的第一个元素 - 明白了上一条,你应该就能理解,为什么我们要在第一个
for
循环里面,第二个for
循环的外面声明largest
。简单来说,我们需要找的是arr[0]
,arr[1]
,arr[2]
... 的最大值。举个例子来说,我们希望largest
在从arr[0]
换成arr[1]
的时候重新赋值 - 在第二层
for
循环结束之后,我们就得到了当前数组的最大值,因此需要把它push
到result
中
- 数组方法中,有个叫
.reduce()
的。之前提到过,是在遍历中,把上一次计算结果用于下一次计算 - 因此,对于上面两个
for
循环,我们都可以通过reduce
来改写。寻找最大值的改写这里不再赘述,请参考找出最长单词那篇"优化"部分,对reduce
的详细解释 - 对于生成结果数组,一样是可以通过
reduce
来实现的。初始值设置为一个空数组[]
,然后把每一个最大值push
进去,记得一定要返回这个数组,用于下次计算。.push()
的返回值不是数组,而是操作之后的数组长度
function largestOfFour(arr) {
return arr.reduce(function(prevArr, nextArr) {
return prevArr.concat(nextArr.reduce(function (prevNum, nextNum) {
return Math.max(prevNum, nextNum);
}, 0));
}, []);
}
- 一定要注意这里的层级关系。我们可以简单理解为,一层
reduce
就相当于一层for
循环。因此,第一层reduce
是遍历传入的参数,也就是二维数组,第二层才是遍历数组中的所有元素,也就是要找出最大值的那些数字 - 理解了上面这一点,也就应该能理解为什么在第一层要传入空数组作为默认值,而在第二层要传入 0 作为默认值
- 简单解释一下每一层发生了什么。里面的那一层,对于一个数组,比如
[1, 2, 3, 4]
来说,设定一个初始值为 0,并作为第一次遍历的prevNum
传入,这时候nextNum
为第一个元素,就是 1。我们通过Math.max()
,返回较大的 1。这时候 1 作为下一次执行的prevNum
传入,同时,nextNum
为第二个元素,就是 2 - 外面那一层,第一次执行,用
[]
作为prevArr
,这时候nextArr
为参数二维数组中的第一个数组。我们计算出nextArr
的最大值,然后把这个结果concat
到prevArr
。之所以用concat
,是因为concat
返回操作后的数组,而push
并没有这个效果 - 如果你还没想明白,要么
console.log
一下每一层执行过程中的prev
和next
,要么对照着找出最长单词这一篇,自己画出来一个表格,应该就能理解了
- 我们可以注意到一点,传入的参数是数组,返回的结果也是数组。进一步思考,传入的参数长度应该与结果的长度是一致的,因为参数长度就表示我们需要找出多少个数组中的最大值
- 如果能想明白上面这一点,那么用
.map()
就顺理成章了。之前提到过,.map()
适用于"返回一个相同长度,并经过一些处理的数组" - 至于找出数组最大值,最简便的写法恐怕是
Math.max.apply(null, arr)
。这一点在找出最长单词那一篇也提到了
function largestOfFour(arr) {
return arr.map(function (e) {
return Math.max.apply(null, e);
});
}
- 这是我能想到的,看起来最简洁的方式
- 只需要想清楚,调用
arr.map()
,这时候,回调函数里面的e
其实是数组,也就是对应着arr[0]
,arr[1]
...