js 中的this指向问题

JS中this上下文对象使用方式

转载  2016-10-09 

这篇文章主要为大家详细介绍了JS中this上下文对象使用方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

JavaScript 有一套完全不同于其它语言的对 this 的处理机制。 在五种不同的情况下 ,this 指向的各不相同。

有句话说得很在理 -- 谁调用它,this就指向谁

一、全局范围内

在全局范围内使用this ,它将指向全局对象(浏览器中为 window)

?
1
2
3
4
5
6
7
8
9
10
11
var name = 'name1' ;
console.log(name);
 
this .name = 'name2' ;
console.log(name);
console.log( this .name);
 
window.name = 'name3' ;
console.log(name);
console.log( this .name);
console.log(window.name);

二、函数调用

直接调用一个函数,this 默认会指向全局 (浏览器端为window)

?
1
2
3
4
5
6
7
8
var name = 'name1' ;
function sayName(){
  console.log(name);
  console.log( this );
}
 
sayName();
window.sayName();

可以看到

还有几个常见的情况,根据谁调用方法就指向谁的原则,this的指向要细看

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
// 全局 name
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
     // 调用它的时候 this指向全局
   return function (){
    console.log( this .name);
   };
  },
  changeName: function (){
     // 调用它的时候 this指向全局
   setTimeout( function (){
    this .name = 'name3' ;
   },0);
  }
};
 
obj.sayName()();
obj.changeName();
setTimeout( function (){
  console.log(name);
  console.log(obj.name);
},0);

像这些类似匿名的函数,默认都是被全局(浏览器下的window)对象调用,要正确地让obj调用,就要指代好

可以用that保持this再进行下一步,或者匿名函数传值,或者使用call/apply/bind改变context等

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
   var that = this ;
   return function (){
    console.log(that.name);
   };
  },
  changeName: function (){
   var that = this ;
   setTimeout( function (){
    that.name = 'name3' ;
   },0);
  }
};
 
obj.sayName()(); // name2
obj.changeName();
setTimeout( function (){
  console.log(name); // name1
  console.log(obj.name); // name3
},0);

三、作为对象方法的调用

其实就类似上头提到的 obj.sayName()  obj.name 等

这时this会指向这个obj 

四、call/apply/bind 的调用

当使用 Function.prototype 上的 call 或者 apply ,bind 方法时,函数内的 this将会被 显式设置为函数调用的第一个参数。

具体使用方法

我们可以稍微修改一下上头的代码,就可以看到this指向的改变

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
   // 返回一个默认全局的函数
   return function (){
    console.log( this .name);
   };
  },
  changeName: function (){
   // 返回一个默认全局的函数
   setTimeout( function (){
    this .name = 'name3' ;
   // 然后将该函数绑定给this(当前obj对象)
   }.bind( this ),0);
  }
};
 
// obj.sayName()这个函数,让obj来调用
obj.sayName().call(obj);
// 让this(也就是全局对象)来调用
obj.sayName().apply( this );
 
obj.changeName();
setTimeout( function (){
  // 输出更改之后,全局name的值
  console.log(name);
  // 输出更改之后,obj对象中 name的值
  console.log(obj.name);
},0);

五、作为构造函数调用

比如 new Foo();

先来看个简单的例子:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
var name = 'name1' ;
function Foo(){
  // 赋值this(当前对象)的name属性值
  this .name = 'name2' ;
}
 
// new 构造函数产生一个实例
var foo = new Foo();
 
console.log(name);
console.log(foo.name);
 
// 直接调用该函数
Foo();
console.log(name);

可以看到,如果函数倾向于和 new 关键词一块使用,则我们称这个函数为构造函数,当new 了之后,this则指向这个心创建的对象(这个new 的过程其实也涉及到了继承机制)。

若直接调用这个函数,this就默认执行全局对象了。

JS中this上下文对象使用方式

转载  2016-10-09   作者:imwtr    我要评论

这篇文章主要为大家详细介绍了JS中this上下文对象使用方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

JavaScript 有一套完全不同于其它语言的对 this 的处理机制。 在五种不同的情况下 ,this 指向的各不相同。

有句话说得很在理 -- 谁调用它,this就指向谁

一、全局范围内

在全局范围内使用this ,它将指向全局对象(浏览器中为 window)

?
1
2
3
4
5
6
7
8
9
10
11
var name = 'name1' ;
console.log(name);
 
this .name = 'name2' ;
console.log(name);
console.log( this .name);
 
window.name = 'name3' ;
console.log(name);
console.log( this .name);
console.log(window.name);

二、函数调用

直接调用一个函数,this 默认会指向全局 (浏览器端为window)

1
2
3
4
5
6
7
8
var name = 'name1' ;
function sayName(){
  console.log(name);
  console.log( this );
}
 
sayName();
window.sayName();

可以看到

还有几个常见的情况,根据谁调用方法就指向谁的原则,this的指向要细看

 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
// 全局 name
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
     // 调用它的时候 this指向全局
   return function (){
    console.log( this .name);
   };
  },
  changeName: function (){
     // 调用它的时候 this指向全局
   setTimeout( function (){
    this .name = 'name3' ;
   },0);
  }
};
 
obj.sayName()();
obj.changeName();
setTimeout( function (){
  console.log(name);
  console.log(obj.name);
},0);

像这些类似匿名的函数,默认都是被全局(浏览器下的window)对象调用,要正确地让obj调用,就要指代好

可以用that保持this再进行下一步,或者匿名函数传值,或者使用call/apply/bind改变context等

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
   var that = this ;
   return function (){
    console.log(that.name);
   };
  },
  changeName: function (){
   var that = this ;
   setTimeout( function (){
    that.name = 'name3' ;
   },0);
  }
};
 
obj.sayName()(); // name2
obj.changeName();
setTimeout( function (){
  console.log(name); // name1
  console.log(obj.name); // name3
},0);

三、作为对象方法的调用

其实就类似上头提到的 obj.sayName()  obj.name 等

这时this会指向这个obj 

四、call/apply/bind 的调用

当使用 Function.prototype 上的 call 或者 apply ,bind 方法时,函数内的 this将会被 显式设置为函数调用的第一个参数。

具体使用方法

我们可以稍微修改一下上头的代码,就可以看到this指向的改变

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
var name = 'name1' ;
 
var obj = {
  name: 'name2' ,
  sayName: function (){
   // 返回一个默认全局的函数
   return function (){
    console.log( this .name);
   };
  },
  changeName: function (){
   // 返回一个默认全局的函数
   setTimeout( function (){
    this .name = 'name3' ;
   // 然后将该函数绑定给this(当前obj对象)
   }.bind( this ),0);
  }
};
 
// obj.sayName()这个函数,让obj来调用
obj.sayName().call(obj);
// 让this(也就是全局对象)来调用
obj.sayName().apply( this );
 
obj.changeName();
setTimeout( function (){
  // 输出更改之后,全局name的值
  console.log(name);
  // 输出更改之后,obj对象中 name的值
  console.log(obj.name);
},0);

五、作为构造函数调用

比如 new Foo();

先来看个简单的例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
var name = 'name1' ;
function Foo(){
  // 赋值this(当前对象)的name属性值
  this .name = 'name2' ;
}
 
// new 构造函数产生一个实例
var foo = new Foo();
 
console.log(name);
console.log(foo.name);
 
// 直接调用该函数
Foo();
console.log(name);

可以看到,如果函数倾向于和 new 关键词一块使用,则我们称这个函数为构造函数,当new 了之后,this则指向这个心创建的对象(这个new 的过程其实也涉及到了继承机制)。

若直接调用这个函数,this就默认执行全局对象了。

转载于:https://www.cnblogs.com/pengwenfan/p/8833094.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值