写作意图
这篇文章用于总结一些javascript语言中常见的易混淆点。
call | apply | bind
在js中,最诡异莫测的莫过于this了,理解的不够深入或是应用场景略微复杂,使用时就会出现各种意想不到的错误。所以,在很多时候,我们需要手动指定上下文环境,来修正this的指向。
this的指向不是函数声明时确定的,而是再函数调用时确定的,最简单判断this所在环境的方法是,寻找this的实际调用者。
一个典型的错误
1 | var a = { |
我们本想将对象a的sayName赋值给变量b,通过调用来查看a的name是什么?结果却输出了undefined。发生这个异常的原因就是因为在调用函数b时,sayName中的this已经不再指向对象a而是指向了全局对象window,由于window下并没有name属性,所以输出undefined。
下面我们将通过以上三种方法来修正这个问题。
使用call
1 | var b = a.sayName; |
使用call方法,第一个参数为函数调用时的上下文环境,将其设定为对象a,这样this就会指向对象a,如此便可以取到对象a中的name属性,输出想要的值。
使用apply
1 | var b = a.sayName; |
使用apply方法,传入上下文环境,可以实现同样的效果。
call和apply的区别
那么call和apply的区别是什么呢?他们的区别在于后续的参数。让我们改造一下上面的代码,来观察效果。
1 | var a = { |
call和apply的区别在于后续参数,call依次传入后续参数,将被函数所使用。apply需要将后续参数以数组的形式传入。
使用bind
bind同样可以用来修正this的指向,它与以上二者的区别在于,bind绑定之后并不会立即执行函数。
1 | var b = a.sayName; |
在为b绑定a的上下文环境之后,b并不会立即执行。
1 | b.bind(a)(); //ein |
如此,便可以如愿得到你想要的效果了。使用bind方法可以延缓函数的执行时间,在你想调用时再执行函数。
bind方法同样可以传入其它参数,和call方法的传入方式相同。
splice | slice | split
看到这三个方法有没有头晕目眩的感觉?因为它们三个实在是太像三胞胎了,真的很难区分。首先,我们来学习或是回顾一下这三个单词。
1 | splice 拼接 |
大多数api其实其名称都与其用途有所关联,这三个api便是很经典的案例。
- splice意为拼接,它的用途,便是将一个数组分割开,并且可以再以指定的方式重新拼接在一起。
- slice意为切片,我们可以使用它来在一个数组或是字符串中切取我们想要的一段。
- split意为分裂,它可以将一个字符串分裂成一个数组。
使用splice
1 | var a = [1,2,3,4,5]; |
splice(starts, count, item1, …, itemx)方法接受多个参数,第一个参数为删除的起始元素,第二个参数为删除数量,后续为插入的内容。
注意: splice方法会修改原始数组,返回被删除的内容数组。
使用slice
1 | var a = [1,2,3,4,5]; |
- slice方法既可应用于数组也可应用于字符串。
- slice(stats, beforeEnds)方法接受最多两个参数,第一个参数代表的序列号必须小于第二个参数,第二个参数为切片的终止位置,第二个参数省略时默认截取到数据末尾,参数为负数时,将反向查找匹配项。
注意: slice方法不会修改原始数组,返回的是被切片节选的片段。
使用split
1 | var a = '123456'; |
split(separator, count)方法可接受两个参数,第一个参数为分割符,用于指定字符串的分割规则,第二个参数为返回数组的最大长度,返回的输出长度不会大于这个参数。
注意: split不会修改原始字符串,返回值为新数组。
map | forEach | reduce | filter | every | some
这六个方法时常用的操作数组的api,均为Array.prototype的本地方法。所以一切数组均可使用这些方法遍历操作数组的每一项,下面将逐一介绍这些方法。
map
1 | var arr = [ |
map(fn(item, index))方法接收一个函数作为参数,这个函数接收两个参数,第一个参数为每一项的数组内容,第二个参数为数组下标。
注意: map方法返回一个新的数组,如果在操作中没有return返回值,默认返回一个值为undefined的数组。
默认返回:[undefined, undefined, undefined]
forEach
1 | arr.forEach((item, index) => { |
forEach(fn(item, index))方法接收一个函数作为参数,这个函数接收两个参数,第一个参数为每一项的数组内容,第二个参数为数组下标。
注意: forEach方法直接操作原始数组,并且不返回任何内容。
reduce
简单用例
1 | var arr1 = [1,2,3,4,5]; |
复杂用例
1 | var res1 = arr1.reduce((curr, next,index,arr) => { |
content | 初始值(curr) | 当前元素(next) | 当前元素索引(index) | 当前元素所属数组(arr) | 函数初始值 |
---|---|---|---|---|---|
content | 10 | 1 | 0 | [1, 2, 3, 4, 5] | 10 |
content | 11 | 2 | 1 | [1, 2, 3, 4, 5] | 10 |
content | 13 | 3 | 2 | [1, 2, 3, 4, 5] | 10 |
content | 16 | 4 | 3 | [1, 2, 3, 4, 5] | 10 |
content | 20 | 5 | 4 | [1, 2, 3, 4, 5] | 10 |
reduce方法用于对数组进行累积化操作,常用于数组求和。接收两个参数,第一个参数为操作函数,第二个参数为函数初始值。对于数组的操作不会修改原始值。
filter
1 | var res = arr1.filter((item, index) => { |
filter方法用于过滤数组的每一项,删选出符合条件的项,并组成一个新的数组。
every
1 | var res = arr1.every((item, index) => { |
every方法用于检查数组的每一项是否符合条件,全部符合条件时返回true,否则返回false。
some
1 | var res = arr1.some((item, index) => { |
some方法用于检查数组中的是否存在符合条件的项,存在则返回true,否则返回false。
另外
一副生动有趣的图解,everyday will be better!