# this 指向
# 核心两要素
this
的指向不是在函数定义时确定的,而是在函数调用时确定this
默认情况下指向window
,严格模式下为undefined
# this 指向分类
# 1️⃣ 隐世绑定
⭐️ 重点: this
指向距离其最近的调用者
所谓的最近的调用者就函数前面最近的一个对象。
function test() {
console.log(this)
}
var obj = {
name: 'Mike',
test: function () {
console.log(this)
}
}
test() // 输出: window
obj.test() // 输出: obj { name: 'Mike', test: f () }
# 2️⃣ 显式绑定
在 JavaScript 中,call
, apply
, bind
可以修改 this
的指向:
- 函数调用
call
,apply
,bind
后,绑定的this
会指向传入的第一个参数 - 如果函数调用
call
,apply
,bind
时没有传入参数,则指向默认对象window
(或undefined
)
function test() {
console.log(this)
}
var obj = {
name: 'Mike'
}
test.call(obj) // obj { name: 'Mike' }
test.apply() // window
test.bind(obj)() // obj { name: 'Mike' }
# 3️⃣ 构造函数中的 this
// 构造函数的 this
function test() {
this.name = 'Rose'
console.log(this)
}
// 返回一个对象类型
function testObj() {
this.name = 'Rose'
return { name: 'Jack' }
}
// 返回一个基本类型
function testBasic() {
this.name = 'Rose'
return 3
}
new test() // 输出: test {name: "Rose"}
console.log(new test()) // 输出: test {name: "Rose"}
console.log(new testObj()) // 输出: {name: "Jack"}
console.log(new testBasic()) // 输出: testBasic {name: "Rose"}
⭐️ 重点: 构造函数中的 this
指向该函数创建的实例对象
⭐️⭐️⭐️ 拓展:
如果该构造函数返回一个对象类型(如Object
, Array
, Function
, Symbol
, null
除外),则通过new
关键字创建的实例指向其返回值,否则返回该构造函数创建的实例对象。
# 4️⃣ 箭头函数中的 this
摘自 MDN — 箭头函数
不只有箭头函数中的 this
指向,更是对箭头函数的总结!!!
MDN: 箭头函数表达式的语法比函数表达式更简洁,并且没有自己的
this
,arguments
,super
或new.target
。箭头函数表达式更适用于那些本来需要匿名函数的地方,并且它不能用作构造函数。
引入箭头函数有两个方面的作用:更简短的函数并且不绑定this。
没有单独的this:在箭头函数出现之前,每一个新函数根据它是被如何调用的来定义这个函数的
this
值:
- 如果是该函数是一个构造函数,
this
指针指向一个新的对象- 在严格模式下的函数调用下,
this
指向undefined
- 如果是该函数是一个对象的方法,则它的
this
指针指向这个对象- 等等
⭐️⭐️⭐️ 重点: 箭头函数不会创建自己的 this
,它只会从自己的作用域链的上一层继承 this
。
⭐️ 与严格模式的关系:
鉴于 this 是词法层面上的,严格模式中与
this
相关的规则都将被忽略。
function foo() {
var f = () => { 'use strict'; return this; };
var g = function () { 'use strict'; return this }
console.log(this) // window 或者 global
console.log(f() === window); // 或者 global,输出 true,箭头函数 f 不受严格模式影响
console.log(g() === window); // 或者 global,输出 false,因为严格模式下 函数 g 的 this 为 undefined
}
foo()
严格模式的其他规则依然不变.
⭐️ 通过 call
或 apply
调用:
由于
箭头函数没有自己的this指针
,通过call()
或apply()
方法调用一个函数时,只能传递参数(不能绑定this
---译者注),他们的第一个参数会被忽略。(这种现象对于bind
方法同样成立---译者注)
var adder = {
base : 1,
add : function(a) {
var f = v => v + this.base;
return f(a);
},
addThruCall: function(a) {
var f = v => v + this.base;
var b = {
base : 2
};
return f.call(b, a);
}
};
console.log(adder.add(1)); // 输出 2
console.log(adder.addThruCall(1)); // 仍然输出 2,说明 箭头函数的 call 的第一个参数被忽略了
⭐️ 不绑定arguments
:
箭头函数不绑定
arguments
对象。因此,在本示例中,arguments
只是引用了封闭作用域内的arguments
:
var arguments = [1, 2, 3];
var arr = () => arguments[0];
arr(); // 1
function foo(n) {
var f = () => arguments[0] + n; // 隐式绑定 foo 函数的 arguments 对象. arguments[0] 是 n,即传给foo函数的第一个参数
return f();
}
foo(1); // 1 + 1 => 2
foo(2); // 2+ 2 => 4
foo(3); // 3+ 3 => 6
foo(3,2);// 4 + 4 => 6
在大多数情况下,使用 剩余参数
是相较使用 arguments
对象的更好选择。
function foo(arg) {
var f = (...args) => args[0];
return f(arg);
}
foo(1); // 1
function foo(arg1,arg2) {
var f = (...args) => args[1];
return f(arg1,arg2);
}
foo(1,2); //2
⭐️ 使用箭头函数作为方法
'use strict';
var obj = {
i: 10,
b: () => console.log(this.i, this),
c: function() {
console.log( this.i, this)
}
}
obj.b(); // undefined, Window{...}
obj.c(); // 10, Object {...}
'use strict';
var obj = {
a: 10
};
Object.defineProperty(obj, "b", {
get: () => {
console.log(this.a, typeof this.a, this);
return this.a+10;
// 代表全局对象 'Window', 因此 'this.a' 返回 'undefined'
}
});
obj.b; // undefined "undefined" Window {postMessage: ƒ, blur: ƒ, focus: ƒ, close: ƒ, frames: Window, …}
结论:箭头函数没有定义this绑定。
⭐️ 使用 new
操作符
箭头函数不能用作构造器,和 new
一起用会抛出错误。
var Foo = () => {};
var foo = new Foo(); // TypeError: Foo is not a constructor
⭐️ 使用 prototype
属性
箭头函数没有 prototype
属性。
var Foo = () => {};
console.log(Foo.prototype); // undefined
⭐️ 使用 yield
关键字
yield
关键字通常不能在箭头函数中使用(除非是嵌套在允许使用的函数内)。因此,箭头函数不能用作函数生成器。
⭐️⭐️⭐️ 综上所述:
- 箭头函数本身不会创建
this
,只会从上级作用域(作用域链)继承this
- 箭头函数的严格模式下,对于this 相关的规则会被忽略,也就是说不会影响其继承上级作用域的
this
。 - 箭头函数不能作为 构造函数,也就是不能与
new
一起使用。 - 箭头函数调用
bind
、call
、apply
时,传入的this
会被忽略。 - 箭头函数没有 原型
prototype
属性。 - 箭头函数没有绑定 arguments,如果在其中使用的话,则是来自上级作用域(函数作用域)的。
# 5️⃣立即执行函数中的 this
立即执行函数中的 this
就一句话:永远指向全局 window
# 6️⃣ DOM 事件处理函数中的 this
摘自 MDN —
this
当函数被用作事件处理函数时,它的 this
指向触发事件的元素(一些浏览器在使用非 addEventListener
的函数动态地添加监听函数时不遵守这个约定)。
// 被调用时,将关联的元素变成蓝色
function bluify(e){
console.log(this === e.currentTarget); // 总是 true
// 当 currentTarget 和 target 是同一个对象时为 true
console.log(this === e.target);
this.style.backgroundColor = '#A5D9F3';
}
// 获取文档中的所有元素的列表
var elements = document.getElementsByTagName('*');
// 将bluify作为元素的点击监听函数,当元素被点击时,就会变成蓝色
for(var i=0 ; i<elements.length ; i++){
elements[i].addEventListener('click', bluify, false);
}
# 7️⃣ 内联事件处理函数中的 this
摘自 MDN —
this
当代码被内联 on-event 处理函数 (opens new window) 调用时,它的 this
指向监听器所在的DOM元素:
<button onclick="alert(this.tagName.toLowerCase());">
Show this
</button>
上面的 alert
会显示 button
。注意只有外层代码中的 this
是这样设置的:
<button onclick="alert((function(){return this})());">
Show inner this
</button>
在这种情况下,没有设置内部函数的 this
,所以它指向 global/window
对象(即非严格模式下调用的函数未设置 this
时指向的默认对象)。
# this 的设计缺陷及其应对方案
1️⃣ 嵌套函数中的 this 不会从外层函数中继承
var obj = {
name: 'Mike',
foo: function () {
console.log(this)
function bar() {
console.log(this)
}
bar()
}
}
obj.foo() // 依次输出 obj、window
上面代码中,foo
函数内部的 bar
函数中的 this
指向的是全局对象 window
,而 foo
函数中的 this
指向的是 obj
对象。
应对方案:
- 在
foo
函数中声明一个变量that
来保存this
,然后在bar
函数中使用that
(其本质是 把this
体系转换为了作用域的体系) - 使用 ES6 中的箭头函数来创建
bar
函数(ES6 中的箭头函数并不会创建其自身的执行上下文,所以箭头函数中的this
取决于它的外部函数)
2️⃣ 普通函数中的 this
默认指向全局对象 window
在默认情况下调用一个函数,其执行上下文中的 this
是默认指向全局对象 window
的。
如果要让函数执行上下文中的 this
指向某个对象,最好的方式是通过 call
、apply
方法来显式调用。
这个问题可以通过为函数设置 JavaScript 的 “严格模式“来解决。因为在严格模式下,默认执行一个函数,其函数的执行上下文中的 this
值是 undefined
。