函数的执行过程中调用位置如何决定this的绑定对象:
1. 默认绑定
全局环境中的this(浏览器环境):
function foo(){
console.log(this)//Window
console.log(this.a)//2
//声明在全局作用域中的变量(比如var a=2)就是全局对象的一个属性,
//它们本质上就是一个东西,并不是通过复制得到的,就像一个硬币的两面一样。
}
var a = 2
foo()//foo()的调用位置
当调用foo()时,this.a被解析成了全局变量a。因为在上例中,函数调用时应用了this的默认绑定,因此this指向全局变量。
那我们怎么知道这里应用了默认绑定呢?可以通过分析调用位置来看看foo()是如何调用的。在代码中,foo()是直接不带任何修饰的函数引用进行调用的。,因此只能使用默认绑定,无法应用其他规则。
如果使用严格模式(strict mode),那么全局对象将无法使用默认绑定,因此this会绑定到undefined:
function foo(){
"use strict"
console.log(this.a)
}
var a = 2
foo()//foo()的调用位置,TypeError: Cannot read property 'a' of undefined
虽然this的绑定规则完全取决于调用位置,但是只有foo()运行在非strict mode下时,默认绑定才能绑定到全局变量;严格模式下与foo()的调用位置无关:
function foo(){
console.log(this.a)
}
var a = 2;//这句要打分号
(function(){
"use strict"
foo();//foo()的调用位置在当前函数里面
//但因为使用了"strict mode",所以this不在当前函数域内绑定
})();//2
通常来说不应该在代码中混合使用strict mode和non-strict mode。整个程序要么严格,要么非严格。然而有时候我们可能会用到第三方库,其严格程度和自己的代码会有所不同,因此一定要注意这类兼容性细节。
2. 隐式绑定
3. 显示绑定
4. new绑定
转载自《你不知道的Javascript》-Kyle Simpson
附:关于箭头函数: