手写数组方法之用于遍历的方法
admin
2024-03-19 02:41:26
0

forEach

用法

可以遍历数组的每个元素,执行特定的操作:

  • 接受两个参数,一个回调函数和一个缺省为 null 的 thisArg。会遍历数组中所有元素,对每个元素执行该函数;
  • 不会新建数组也不会修改原数组,总是返回 undefined(即使明确指定了返回值)
  • 遍历范围在执行回调前就确定了,遍历过程中 push 会修改原数组,但是不会影响遍历范围
  • 遍历过程中可以提前修改后面才会遍历到的元素
  • 会直接跳过 empty 元素
// res 为 undefined
const res = ['a','b','c'].forEach(function(item,index,arr){console.log(`${index}-${item}`,arr,this)return 123
},{})                                                        
// 遍历过程中即使push了新元素,也仍然按照原数组长度进行遍历。打印 1,2
[1,2].forEach((item,index,arr) => {arr.push(3,4,5)console.log(item)
})
// 遍历过程中可以提前修改未遍历元素。打印 1,100
[1,2].forEach((item,index,arr) => {arr[1] = '100'console.log(item)
})
// 遍历过程中可以 return,但只是结束当前这次遍历,无法跳出整个 forEach。打印 2
[1,2].forEach((item,index,arr) => {if(index == 0) returnconsole.log(item)
})
// 遍历过程中会自动跳过 empty 元素(null 和 undefined 不会跳过)。打印 1,2,4
[1,2,,4].forEach((item,index,arr) => {    console.log(item)
})

实现

Array.prototype.myforEach = function (fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = this    for(let i = 0,len = arr.length;i < len;i++){// 如果不是 empty 元素if(i in arr){fn.call(thisArg,arr[i],i,arr)}}
}

map

用法

将原数组的每个元素映射为执行回调函数之后的返回值

  • 基本实现和 forEach 差不多,也是会跳过 empty 元素
  • forEach 的遍历范围一开始就确定好,所以需要先保存最初数组长度;同理,map 最终返回的新数组长度也是一开始就与原数组长度绑定好了,声明新数组的时候需要指定这个长度
// 返回新数组 [2,4,6,8]
[1,2,3,4].map((item,index) => item * 2)

实现

Array.prototype.myMap = function(fn,thisArg = null){if(typeof fn != 'function'){throw TypeError(`${fn} is not a function`)}let arr = this// 这里不要使用 let newArr = [],否则修改原数组长度时会影响新数组长度let newArr = new Array(arr.length)for(let i = 0,len = arr.length;i < len;i++){if(i in arr){const res = fn.call(thisArg,arr[i],i,arr)newArr[i] = res}}return newArr
}

flatMap

用法

flatMap 相当于是 map 和 flat(1) 的结合。它会给某个数组调用 map 方法,如果得到了一个多维数组,则会对该数组进行一次降维

PS:注意这个方法不会改变原数组。

const arr1 = [1, 2, 3, 4];arr1.map(x => [x * 2]);
// [[2], [4], [6], [8]]arr1.flatMap(x => [x * 2]);
// [2, 4, 6, 8]

实现

我们可以在每次执行 flatMap 的回调并返回一个新结果时,判断该结果是不是数组,如果是则取出数组中的每个元素放入最终返回的新数组中。

Array.prototype.myFlatMap = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thislet newArr = new Array(arr.length)let k = 0for(let i = 0;i < arr.length;i++){if(i in arr){const res = fn.call(thisArg,arr[i],i,arr)if(Array.isArray(res)){for(let el of res){newArr[k++] = el}} else {newArr[k++] = res}}}   return newArr
}

find

用法

find 返回数组中第一个符合条件的元素或者 undefined:

  • 通过indexOf 搜索数组,无法自定义搜索条件,所以出现了 find
  • find 会对每个元素执行一次回调函数,直到找到符合条件的元素,就将这个元素返回(永远只返回一个),并结束函数执行;找不到则返回 undefined
  • 注意这个方法不会跳过 empty 元素,所以这里不做 i in arr 的检查
[1,2,3,4,5].find((item,index,arr) => item > 2)      // 3

实现

Array.prototype.myFind = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thisfor(let i = 0;i < arr.length;i++){const result = fn.call(thisArg,arr[i],i,arr)if(result){return arr[i]}}return undefined
}

findIndex

用法

和 find 基本一致,但是 findIndex 返回的是第一个符合条件的元素的索引,没有这样的元素就返回 -1

[1,2,3,4,5].findIndex((item,index) => item > 2)      // 2

实现

Array.prototype.myfindIndex = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thisfor(let i = 0;i < arr.length;i++){const result = fn.call(thisArg,arr[i],i,arr)if(result){return i}}return -1
}

filter

用法

通过 find 搜索数组,只能找到一个符合条件的元素,而 filter 可以筛选出所有符合条件的元素:

[1,2,3,4,5].filter((item,index) => item > 2)        // 返回 [3,4,5]
[1,2,3,4,5].filter((item,index) => item > 100)      // 没有符合的元素,返回 []

实现

Array.prototype.myFilter = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thislet res = []let k = 0for(let i = 0;i < arr.length;i++){if(i in arr){const result = fn.call(thisArg,arr[i],i,arr)// 如果元素符合条件,则放入新数组中if(result){res[k++] = arr[i]}}}
}

some

用法

接受一个回调函数表示判断条件,只要数组中有一个元素满足该条件(回调函数返回 true),some 方法就返回 true,否则返回 false

[1,2,3,4].some((item,index) => item>3)      // 至少有一个大于3的数,返回 true
[1,2,3,4].some((item,index) => item>100)    // 没有一个大于100的数,返回 false

实现

Array.prototype.mySome = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thisfor(let i = 0;i < arr.length;i++){const result = fn.call(thisArg,arr[i],i,arr)if(result){return true}}return false
}

every

用法

接受一个回调函数表示判断条件,只有数组中所有元素都满足该条件(回调函数返回 true),every 方法才会返回 true,有一个不满足都会返回 false

[1,2,3,4].every((item,index) => item>0)     // 所有元素都大于0,返回 true
[1,2,3,4].every((item,index) => item>3)     // 并非所有元素都大于3,返回 false

实现

Array.prototype.myEvery = function(fn,thisArg = null){if(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}let arr = thisfor(let i = 0;i < arr.length;i++){const result = fn.call(thisArg,arr[i],i,arr)if(!result){return false}}return true
}

reduce

用法

reduce 可以归并数组的每个元素,最终构建一个累计归并值作为返回值:

  • 语法:arr.reduce((acc,cur,index,arr) => {...},baseAcc)
  • 接受两个参数,一个是回调函数,一个是初始累计归并值。其中,回调的当前返回值会作为下次迭代所使用的 acc,也即累计归并值。初始累计归并值缺省是第一个非 empty 元素,且此时会从该元素的下一个元素开始迭代。
// 没有提供初始累计归并值,因此缺省是1,并且从2开始迭代
[1,2,3,4].reduce((acc,cur) => acc + cur)              // 10// 提供100作为初始累计归并值,从1开始迭代
[1,2,3,4].reduce((acc,cur) => acc + cur,100)          // 110// 二维数组转化为一维数组,concat 本身会拍平一维数组
[1,2,[3,4]].reduce((acc,cur) => acc.concat(cur),[])       // [1,2,3,4]

实现

实现的时候,有两个关键的地方:

  • 如何判断有没有传第二个参数?用 typeof baseAcc === 'undefined' 判断不准确,因为有可能传的第二个参数确实就是 undefined,这里可以通过剩余参数的长度判断
  • 如何找出数组的第一个非 empty 元素?遍历数组,用 in 判断
Array.prototype.myReduce = function(...args){let fn = args[0]let arr = thislet len = arr.lengthlet index = 0,accif(typeof fn != 'function'){throw new TypeError(`${fn} is not a function`)}// 如果传了第二个参数if(args.length >= 2){acc = args[1]} else {// 只要当前数组还没找到非 empty 元素,就一直遍历下去while(index < len && !(index in arr)){index++}// 如果数组是一个充满 empty 元素的空数组,则抛出错误if(index >= len){throw new TypeError('Reduce of empty array with no initial value')}// index 加一,表示第一个非 empty 元素的下一个元素acc = arr[index++]for(;index < len;index++){if(index in arr){acc = fn(acc,arr[index],index,arr)}}return acc}    
}

reduceRight

用法

用法基本和 reduce 一致,区别是它是从后往前去遍历数组的。

[0, 1, 2, 3].reduceRight((acc, cur) => {console.log(cur);
});
// 2
// 1
// 0

实现

reduceRight 的实现和 reduce 基本一样,但需要注意:非 empty 元素的查找以及数组的遍历顺序是反过来的

Array.prototype.myReduceRight = function(...args){let fn = args[0]let arr = thislet len = arr.lengthlet index = len - 1,accif(typeof fn != 'function'){throw new TypeError(`${fn} is not function`)}if(args.length >= 2){acc = args[1]} else {while(index > 0 && !(index in arr)){index--}if(index == 0){throw new TypeError('Reduce of empty array with no initical value')}acc = arr[index--]for(;index >= 0;index--){if(index in arr){acc = fn(acc,arr[index],index,arr)}}return acc}
}

相关内容

热门资讯

linux入门---制作进度条 了解缓冲区 我们首先来看看下面的操作: 我们首先创建了一个文件并在这个文件里面添加了...
C++ 机房预约系统(六):学... 8、 学生模块 8.1 学生子菜单、登录和注销 实现步骤: 在Student.cpp的...
A.机器学习入门算法(三):基... 机器学习算法(三):K近邻(k-nearest neigh...
数字温湿度传感器DHT11模块... 模块实例https://blog.csdn.net/qq_38393591/article/deta...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
Redis 所有支持的数据结构... Redis 是一种开源的基于键值对存储的 NoSQL 数据库,支持多种数据结构。以下是...
win下pytorch安装—c... 安装目录一、cuda安装1.1、cuda版本选择1.2、下载安装二、cudnn安装三、pytorch...
MySQL基础-多表查询 文章目录MySQL基础-多表查询一、案例及引入1、基础概念2、笛卡尔积的理解二、多表查询的分类1、等...
keil调试专题篇 调试的前提是需要连接调试器比如STLINK。 然后点击菜单或者快捷图标均可进入调试模式。 如果前面...
MATLAB | 全网最详细网... 一篇超超超长,超超超全面网络图绘制教程,本篇基本能讲清楚所有绘制要点&#...
IHome主页 - 让你的浏览... 随着互联网的发展,人们越来越离不开浏览器了。每天上班、学习、娱乐,浏览器...
TCP 协议 一、TCP 协议概念 TCP即传输控制协议(Transmission Control ...
营业执照的经营范围有哪些 营业执照的经营范围有哪些 经营范围是指企业可以从事的生产经营与服务项目,是进行公司注册...
C++ 可变体(variant... 一、可变体(variant) 基础用法 Union的问题: 无法知道当前使用的类型是什...
血压计语音芯片,电子医疗设备声... 语音电子血压计是带有语音提示功能的电子血压计,测量前至测量结果全程语音播报࿰...
MySQL OCP888题解0... 文章目录1、原题1.1、英文原题1.2、答案2、题目解析2.1、题干解析2.2、选项解析3、知识点3...
【2023-Pytorch-检... (肆十二想说的一些话)Yolo这个系列我们已经更新了大概一年的时间,现在基本的流程也走走通了,包含数...
实战项目:保险行业用户分类 这里写目录标题1、项目介绍1.1 行业背景1.2 数据介绍2、代码实现导入数据探索数据处理列标签名异...
记录--我在前端干工地(thr... 这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 前段时间接触了Th...
43 openEuler搭建A... 文章目录43 openEuler搭建Apache服务器-配置文件说明和管理模块43.1 配置文件说明...