JS: 数组的循环函数
JS 数组相关的循环函数,用得挺多,所以有些坑还是要去踩一下,先来看一道面试题。
注意:下面提到的不改变原数组仅针对基本数据类型。
面试题
模拟实现数组的 map 函数。
心中有答案了吗?我的答案放在最后。
map( callback( cur, index, arr ), thisArg )
map 方法创建一个新数组,其结果是该数组中的每个元素都调用一个提供的函数后返回的结果。
map 不改变原数组(可以在 callback 执行时改变原数组)
let a = [1, 2, 3]
let b = a.map(item => item*2) // 不改变原数组
a // [1, 2, 3]
b // [2, 4, 6] // 在 callback 执行时改变原数组
a.map((cur, i, arr) => arr[i] *= 2)
a // [2, 4, 6]
没有 return 时,返回
undefinedlet c = a.map((cur, i, arr) => arr[i] *= 2) c // [undefined, undefined, undefined]
原数组中新增加的元素不会被 callback 访问
let a = [1, 2, 3]
let b = a.map((cur, i, arr) => {
arr.push(arr.length + 1)
return cur*2
}) b // [2, 4, 6]
a // [1, 2, 3, 4, 5, 6]
filter( callback( cur, index, arr ), thisArg )
filter 方法创建一个新数组, 其包含通过所提供函数实现的测试的所有元素。
这个没什么好说的,filter 只返回过滤后的新数组,依然不会改变原数组
let a = [1, 2, 3]
let b = a.filter(item => item % 2 == 0)
a // [1, 2, 3]
b // [2]
需要注意的是,尽量不要在 filter 中直接return item。
/* 一般情况是没问题的 */
let a = [1, 2, 3]
let b = a.filter(item => item)
a // [1, 2, 3]
b // [1, 2, 3]
/* false、0、undefined、null */
let c = [true, false, 0, undefined, null]
let d = c.filter(item => item)
c // [true, false, 0, undefined, null]
d // [true]
一般情况下直接return item是没问题的,但遇到false、0、undefined、null这几个特殊值时,就会出问题。因为 filter 是根据return的值来判断返回的新数组是否要添加遍历到的原数组索引值,而不是直接在新数组中添加return的值。
简单来说,就是 filter 中的 callback 的 return 应该返回一个 boolean 值,true 即满足过滤条件,false 则不满足过滤条件。
forEach( callback( cur, index, arr ), thisArg )
forEach 方法对数组的每个元素执行一次提供的函数。
forEach 无法中断,需要中断的则表明:不应该使用 forEach。当然,使用 try、catch 是可以实现中断的,但不推荐。
let a = [1, 2, 3] try {
a.forEach(item => {
console.log(item)
if (item % 2 == 0) throw new Error('hhh')
})
} catch (e) {
console.log('中断')
} /*
1
2
中断
*/
forEach 也是不会改变原数组的
let a = [1, 2, 3]
a.forEach(item => item*2)
a // [1, 2, 3]
every( callback( cur, index, arr ), thisArg )
every 方法测试数组的所有元素是否都通过了指定函数的测试。
这个没什么好说的,但 every 和 some 都有个坑。
坑:空数组调用 every 会返回true。
// 返回了 true
[].every(item => item > 0) // true
some( callback( cur, index, arr ), thisArg )
some 方法测试是否至少有一个元素通过由提供的函数实现的测试。
坑:空数组调用 some 会返回false。
// 和 every 相反,这里返回了 false
[].some(item => item > 0) // false
reduce( callback( prev, cur, index, arr ), initVal )
reduce 方法对数组中的每个元素执行一个由您提供的reducer函数(升序执行),将其结果汇总为单个返回值。
如果没有提供
initVal,reduce 会取数组中的第一个值为prev, 然后从数组索引 1 开始执行 callback,跳过第一个索引。如果提供initlVal,从索引 0 开始,prev为initVal。如果数组为空且没有提供
initVal会报错。[].reduce((prev, cur)=> cur) // TypeError
实现 map 函数
Array.prototype._map = function(callback, thisArg) {
if (typeof callback !== 'function') {
throw new TypeError(`${callback} is not a function`)
}
let res = []
for (let i=0; i<this.length; i++) {
res.push(callback.call(thisArg, this[i], i, this))
}
return res
}
JS: 数组的循环函数的更多相关文章
- js数组的splice函数
一直没搞懂数组的splice函数,今天稍微测试了一下,了解了它的功能,在这里记录一下 1.测试 测试① var a = [1,2,3]; console.info(a.splice(1,1)); co ...
- JS数组(Array)处理函数总结
1.concat() 连接两个或更多的数组该方法不会改变现有的数组,而仅仅会返回被连接数组的一个副本.例如: <script type="text/javascript"&g ...
- js数组之可变函数
在js的数组中有两个方法为数组添加元素:1.push();2.unshift(),push函数是将元素添加到数组的末尾,现在不用说大家估计也能猜出来,unshift这个函数就是把元素添加到数组的开头的 ...
- js数组,数字函数,字符串函数,表单验证,hashMap,堆栈,日期函数,call函数
1.javascript的数组API Js代码 收藏代码 //定义数组 var pageIds = new Array(); pageIds.push('A'); 数组长度 pageIds.lengt ...
- js数组之sort()函数
一般我们使用sort函数进行数组的排序,sort()方法有一个可选参数,是用来确定元素顺序的函数.如果这个参数被省略,那么数组中的元素将按照ASCII字符顺序进行排序.如: var arr = [&q ...
- js 数组的forEach 函数
var numbers = [4, 9, 16, 25]; function myFunction(item, index) { console.log("item:" + ite ...
- js数组 函数
js数组 filter(),map(),some(),every(),forEach(),lastIndexOf(),indexOf() 文章1:http://www.jb51.net/article ...
- js 数组常用的操作函数整理
平时多做企业应用开发,抱着实用为主,对前端技术理解得比较肤浅,下面就是肤浅地对 js 数组的属性和方法及对它操作的 jquery 方法做些记录: js 数组是 js 内建的一个非常强大数据类型,由于 ...
- 页面循环绑定(变量污染问题),js面向对象编程(对象属性增删改查),js字符串操作,js数组操作
页面循环绑定(变量污染问题) var lis = document.querySelectorAll(".ul li") for ( var i = 0 ; i < lis. ...
随机推荐
- 2019.01.02 poj3046 Ant Counting(生成函数+dp)
传送门 生成函数基础题. 题意:给出nnn个数以及它们的数量,求从所有数中选出i∣i∈[L,R]i|i\in[L,R]i∣i∈[L,R]个数来可能组成的集合的数量. 直接构造生成函数然后乘起来f(x) ...
- 2018.12.31 NOIP训练 czy的后宫5(树形dp)
传送门 题意:给一棵有根树,树有点权,最多选出mmm个点,如果要选一个点必须先选其祖先,问选出来的点权和最大值是多少. 直接背包转移就行了. 代码
- 2018.11.14 uoj#34. 多项式乘法(fft)
传送门 NOIpNOIpNOIp爆炸不能阻止我搞oioioi的决心 信息技术课进行一点康复训练. fftfftfft板题. 代码: #include<bits/stdc++.h> usin ...
- 2018.11.05 bzoj3124: [Sdoi2013]直径(树形dp)
传送门 一道sbsbsb树形dpdpdp 第一问直接求树的直径. 考虑第二问问的边肯定在同一条直径上均是连续的. 因此我们将直径记下来. 然后对于直径上的每一个点,dpdpdp出以这个点为根的子树中不 ...
- JAVA折腾微信公众平台(Token验证)[转]
JAVA折腾微信公众平台(Token验证) BAE的JAVA还在内测的时候,抱着好奇的态度发邮件申请了内测权限,当时折腾了一天,然后就没折腾了.现在BAE的JAVA都已经正式开放使用了,我又蛋疼的想写 ...
- 指令发布中如何实现new新消息的提醒?
设计思路:反馈后,最急需了解反馈结果的是申请人,故给每一条反馈信息添加一个查看状态的字段,如CK_STATUS,并为这个状态设计为char(1)类型,java bean中使用integer可以实现默认 ...
- springboot深入学习(一)-----springboot核心、配置文件加载、日志配置
一.@SpringBootApplication @SpringBootApplication是spring boot的核心注解,源码如下: 相当于:@Configuration+@EnableAut ...
- Mysql中Left Join Right Join Inner Join where条件的比较
建立一对多的表 company 和 employee company表 id name address 1baidu北京 2huawei深圳 3jingdong北京 4tengxu ...
- 上传文件夹+php
最近公司做工程项目,实现文件夹上传 网上找了很久,发现网上很多代码大都存在很多问题,不过还是让我找到了一个符合要求的项目. 对项目的文件夹上传功能做出分析,找出文件夹上传的原理,对文件夹的传输模式深入 ...
- wchar_t,char,string,wstring等的总结
一.LPSTR LPCSTR LPTSTR LPCTSTR等 确定的类型: LPSTR = CHAR * = char * LPCSTR = const CHAR * = char * //c意为co ...