This in JavaScript

时间:2024-07-28 08:32:50

声明 本文仅为读书笔记并致力于理解Js中的this关键字。如有雷同,纯属巧合。原因有二,其一:有幸看了同一本书,其二:this这玩意已被说烂了~

  1. 为什么要用this?

    this提供了优雅的方式隐式传递对象的引用

    上实例:

    function sayEng(){
    console.log(this.name+":sayEng");
    }
    function sayChi(){
    console.log(this.name+":sayChi");
    }
    function stu(){
    name = "stu";
    }
    function tea(){
    name = "tea";
    }
    sayEng.call(stu);//stu:sayEng
    sayEng.call(tea);//tea:sayEng
    sayChi.call(stu);//stu:sayChi
    sayChi.call(tea);//tea:sayChi

    当不使用this时

    function sayEng(ctx){
console.log(ctx.name+"sayEng");
}
function sayChi(ctx){
console.log(ctx.name+"sayChi");
}
function stu(){
name = "stu";
}
function tea(){
name = "tea";
}
sayEng(stu);//stu:sayEng
sayEng(tea);//tea:sayEng
sayChi(stu);//stu:sayChi
sayChi(tea);//stea:sayChi
```
两种对比,可以看出当不使用this时,需**手动显式传递一个上下文对象**。而使用了this可以使得代码很精炼。而且传递上下文的方式会使得代码越来越混乱。
2. 对this的误解
1)指向自身(书中代码)
```javascript
function foo(num) {
console.log( "foo: " + num );
// count用于记录foo的被调用次数
this.count++;
}
foo.count = 0;
var i;
for (i=0; i<10; i++) {
if (i > 5) {
foo( i );
}
}
console.log( foo.count ); // 0 -- ??
```
**为什么是0,而不是4?原因在于,此时调用foo函数的是window,this的指向是window。**
解决办法有两种:
其一,用foo代替this。也就是foo.count++。
其二,强制this指向foo,通过call,或者apply。也就是在if里面改为foo.call(foo,i)。
2)指向函数的作用域 ```javascript
function foo() {
var a = 2;
this.bar();
}
function bar() {
console.log( this.a );
}
foo(); //ReferenceError:a is not defined
```
**原因在于:调用foo的对象为window**
3. 绑定规则
1)默认绑定
若函数是直接使用不带任何修饰的函数引用的,它便是应用了默认绑定。this指向window,上面的实例可以佐证。
p.s *当使用严格模式时,则不能将全局对象用于默认绑定*
2)隐式绑定 ```javascript
function foo() {
console.log( this.a );
}
var obj = {
a: 2,
foo: foo
};
obj.foo(); // 2
```
本例就使用了默认绑定,this绑定在obj上。**另外,对象属性引用链只有上一层或者说最后一层在调用位置中起作用。**也即,obj1.obj2.foo(),this绑定在obj2中。
但是,隐式绑定经常会有一个问题产生--**隐式丢失**
```javascript
function foo() {
console.log( this.a );
}
var obj = {
a: 2,
foo: foo
};
var bar = obj.foo; // 函数别名
var a = "oops,global";
bar(); // "oops,globa"
```
this的指向不再是obj,而变成了window。丢失了this的指向。同理还有当obj.foo作为参数传递时丢失,原因在于参数传递相当于两个步骤,一赋值,二参数传递。
3)显示传递
通过call、apply的方式,将this绑定在对象上。具体call、apply的使用方法见本人博客[Call and Apply in JavaScript](http://www.cnblogs.com/zhaoww/p/5247827.html)
4)new绑定
```javascript
function foo(a) {
this.a = a;
}
var bar = new foo( 2 );
console.log( bar.a );// 2
```
使用new调用时,会构造一个对象并将this指向进行绑定。如本例的foo上。
4. 优先级
new绑定>显示绑定>隐式绑定>默认绑定 ----------
>*本人较懒,部分代码原封不动摘抄自原著。还请见谅*