内容总览
一.直接调用(独立函数调用/全局调用)
😂直接调用顾名思义就是直接进行函数的调用,怎么直接调用哪? 我们常见的直接调用分为两种情况,如下:
1 | scss复制代码function foo () { |
这是最常见的一种情况,这种情况this在非严格模式的情况下指向的是全局对象,这种调用方式,我们称之为独立函数调用,或者称之为全局调用。
还有另外一种情况是在对象中定义,然后赋值给一个变量,然后单独对这个赋值后的函数进行调用,依然是独立函数调用或者称之为全局调用,非严格模式下指向的是window。
1 | js复制代码let obj = { |
这种写法,依然属于独立函数调用,指向的依然是window或者说全局对象。
🚨注意:独立函数调用(全局调用)this的指向在严格模式下并不指向window而是指向undefined所以我们在使用的时候千万不要使用this来代替window,而是直接使用window。
二.对象绑定
😈通过对象绑定比较容易理解,因为我们在平时会经常用到,当我们在如下的这种方式进行的时候会指向调用它的对象obj;
1 | js复制代码let obj={ |
三.new绑定
🦊在讲解new绑定之前,我们先来看下当我们在new一个对象的时候到底做了什么事情。
- 创建新的空对象。
- 将this指向这个空对象。
- 指向函数体中的代码。
- 没有显示返回空对象的时候默认返回这个对象。
😶🌫️从第二步我们知道,当我们进行new操作的时候会将this绑定到实例化的这个对象上面。
1 | js复制代码function foo (name) { |
通过上述的代码我们可以看到当我们进行对象实例化的时候函数内部的this指向的是实例化的这个对象。
四.this指向绑定事件的元素
1 | js复制代码<ul id="color-list"> |
1 | js复制代码// this 是绑定事件的元素 |
🤡有些时候我们会遇到一些困扰,比如在div节点的事件函数内部,有一个局部的 callback 方法,该方法被作为普通函数调用时,callback 内部的this是指向全局对象 window的.
1 | html复制代码<div id="div1">我是一个div</div> |
1 | js复制代码window.id = 'window'; |
此时有一种简单的解决方案,可以用一个变量保存 div节点的引用,如下:
1 | js复制代码window.id = 'window'; |
五.显式绑定(call/apply)
🥴有的时候this的指向并不能如我们所愿,这个时候我们需要手动去更改this的指向,来满足我们的需求,其实在JavaScript中给我们提供了能够更改this的绑定,首先我们看下call和apply。
1 | js复制代码function foo(name,age){ |
通过call函数我们可以看到我们可以手动的将this绑定到我们新定义的对象上面来,并且通过call单个传参的方式将参数传递给了这个函数,实现了函数的调用和this的绑定。
1 | js复制代码function foo (name, age) { |
我们会发现我们使用apply的方式进行绑定的修改结果依然如此,差别在于他们的传参方式不同,call是单个的方式进行传参的,而apply是通过数组的方式传参的。
六.bind函数的显式绑定
🐻bind的绑定和call和apply的使用有些差别,使用bind会生成一个新的函数,这个新函数我们称之为BF绑定函数,我们需要手动对这个函数进行调用。
1 | js复制代码function foo(){ |
七.内置函数的调用绑定思考
😂我们在开发中会用到很多内置的函数,比如定时器setTimeout 这个时候我们需要靠经验来判断当前的this指向因为有些东西根本不是我们来调用的,而是函数内部调用的,我们根本不知道他们做了什么,这些内容需要我们自己总结一下,内容如下。
- 定时器内部的函数this指向window
1 | js复制代码setTimeout(function () { |
- 按钮的点击事件,指向事件发起的对象,也就是绑定事件的元素。
1 | js复制代码let box = document.querySelector(".test") |
- forEach中的this也是指向window
1 | js复制代码const array = [1, 2, 3, 4, 5, 6] |
八.绑定优先级的比较
🥴我们前面了解的都是一些独立的规则,但是实际的情况往往是比较复杂的,可能涉及到多个绑定一起使用的情况,这个时候我们就需要研究一下不同函数之间调用的优先级。
- 直接调用(默认绑定)的优先级是最低的。
- 显式绑定优先级高于对象绑定(隐式绑定)。
1 | js复制代码function foo () { |
- new绑定优先级高于对象绑定(隐式绑定)的优先级
1 | js复制代码function foo (name) { |
- new绑定不能和call与apply一起使用,new绑定的优先级比bind高。
1 | js复制代码function foo (name) { |
- bind和apply的优先级,bind的优先级更高,也高于call因为call和apply使用方法一样。
1 | js复制代码function foo () { |
九.绑定规则之外
🥴其实在上述的绑定规则之外还有许多我们有时候按照规则难以理解的情况,我们来总结下有哪些情况。
- 在显式绑定当中如果传入null/undefined这个绑定会被忽略,使用默认规则,严格模式能够绑定。
1 | js复制代码function foo () { |
- 创建一个函数的间接引用,使用函数的默认绑定规则,指向window(了解)
1 | js复制代码var obj1 = { |
十.箭头函数的使用
😂我们之前使用函数的方式是这样的。
1 | js复制代码// 普通函数方式 |
🐣箭头函数的写法
1 | js复制代码// 箭头函数的完整写法 |
😙箭头函数与普通函数的区别:箭头函数中没有this和arguments并且不能作为构造函数使用。
🤡箭头函数的优化方式:
- 当一个参数的时候可以省略函数参数的小括号。
1 | js复制代码let name = ["abc","bca","nba"]; |
- 如果函数体中只有一行执行代码{}可以省略,但是一行代码中不能带return;
1 | js复制代码let name = ["a","b","c"]; |
- 如果函数体中只有一行代码,那么这行代码的返回值会作为整个函数的返回值。
1 | js复制代码let name = ["a","b","c"]; |
- 如果默认返回值是一个对象,那么这个对象必须加小括号
1 | js复制代码let arr = ()=>({name:"zzz",age:12}) |
💗箭头函数使用案例:使用所有nums的所有平方和的值。
1 | js复制代码let nums = [20,30,11,15,111] |
1 | js复制代码/* |
十一.箭头函数中的this
😁箭头函数中是没有this的,所以this如果在箭头函数中的this就是上级作用域的this。
1 | js复制代码let bar =()=>{ |
👽因为没有this的原因,箭头函数中使用显式绑定也是无法绑定过去的。
1 | js复制代码let bar =()=>{ |
😭我们再来看下一个案例来明白下this的查找规则。
1 | js复制代码let obj = { |
- 首先调用foo函数返回bar定义的函数,箭头函数没有this。
- 根据作用域的查找规则,会向上层找this,foo函数中是有作用域的。
- 使用call函数是无法更改掉this的。
- 所以打印出来this的指向是obj函数。
十二.this常见面试题解析
1 | js复制代码const o1 = { |
👽解析:首先o1.fn是一个对象调用,所以this指向的是这个对象o1,o2.fn()单调用的时候返回的是o1.fn因此指向的仍然是o1,第三个o1.fn进行了重新赋值然后调用,独立函数调用指向undefined。
1 | js复制代码var name = "window"; |
1 | js复制代码var name = 'window' |
1 | js复制代码var name = "window" |
1 | js复制代码var name = "window" |
本文转载自: 掘金