this
是JavaScript中的一个重要的概念,它的值并不是由声明的位置来决定的,而是由调用的位置来决定。正是因为这个原因在不同的语境下它的值是不一样的,所以它成了面试官最喜欢考察的内容。
通常this
由下面这四条法则就可以确定:
new绑定,指向创建的对象。
1
2
3
4
5function Student(name){
this.name = name;
}
var s = new Student("小明");
console.log(s.name);//小明通过
new
关键字创建的对象,那么构造方法中的this
指向该对象。call、apply、bind等显式绑定。
1
2
3
4
5
6
7
8
9
10
11function sayName(){
console.log(this.name);
}
var obj1 = {
name : "小李"
};
var obj2 = {
name : "小秋"
};
sayName.call(obj1);//小李
sayName.call(obj2);//小秋拥有上下文对象的隐式绑定。
1
2
3
4
5
6
7var obj = {
name : "小秋",
sayName : function (){
console.log(this.name);
}
};
obj.sayName();//小秋默认绑定:严格模式下绑定
undefined
,非严格模式下绑定全局变量(浏览器环境下是window
)。1
2
3
4
5function sayName(){
console.log(this.name);
}
var name = "小李";
sayName();//小李
this
的绑定也是有优先级的,它的优先级也是按照上面4条从1到4的。没错,你没有看错new绑定比显式绑定的优先级还要高,虽然这种情况并不多见,这里给出一个简单的例子,供大家参考:
1 | function Student(name){ |
上面这4条规则适用于大量的情况,但是俗话说的好“凡是都有例外”,下面2条特殊情况要格外注意。
- 显示绑定(call、apply、bind)如果绑定的是
null
或者undefined
,在严格模式下会绑定对应的值,在非严格模式下绑定全局变量(浏览器环境下是window
)。由于绑定1
2
3
4
5
6
7
8
9
10function sayName(){
console.log(this.name);
}
var obj = {
name : "小李",
sayName :sayName
}
var name = "小秋";
obj.sayName.call(null);//小秋
sayName.call();//小秋null
或者undefined
在非严格模式下会绑定全局变量window
这样会很危险。如果这里非要绑定一个对象但是又不绑定全局变量可以绑定一个空对象就比如{}
或者Object.create(null)
,两者的区别是后者不会创建Object.prototype
这个委托,也就是后者比前者更空。 - 间接引用,本质上是默认绑定。 JS中每一个表达式都有一个返回值,上面
1
2
3
4
5
6
7
8
9
10
11
12function sayName(){
console.log(this.name);
};
var obj1 = {
name : "小李",
sayName : sayName
}
var obj2 = {
name : "小秋"
}
var name = "小芹";
(obj2.sayName = obj1.sayName)();//"小芹"obj2.sayName = obj1.sayName
也是表达式,返回值是函数sayName
的引用。而把引用括起来是为了保证这是一个整体,后面的括号是函数的调用。也就是说这里相当于是(sayName)()
,进一步相当于sayName()
,很显然相当于是window
来调用的。 - 箭头函数,根据所处的环境(作用域)来决定。
1
2
3
4
5
6
7
8
9
10
11
12
13var sayName = (function (){
var name = "小明"
//立即执行函数 这个地方的this是window
return ()=>{
console.log(this.name);
};//箭头函数中的this由所处的作用域来决定 也就是window
})();
var name = "小华";
var obj = {
name:"小芳"
};
sayName();//小华
sayName.call(obj);//sayName对应的箭头函数所处的作用域已指向window 即使call也不能改变指向 故打印小华
其他情况:
- HTML事件中的this指向该DOM元素。
1
2
3
4
5
6<button onclick="handleClick(this)">点我</button>
<script>
function handleClick(obj){
obj.innerHTML = "点击了";//按钮的文案会修改为 点击了
}
</script> - JQuery获取到的DOM元素,
this
指向该DOM元素,$(this)
是该DOM对象的JQuery封装对象。1
2
3
4$("#id").click(function() {
console.log(this);//选中的DOM元素
console.log($(this));//选中DOM元素的JQuery封装对象
});