如何在JavaScript各种各样的上下文中确定this指的是什么?

让我们从下面 Bob 的例子开始:
Js代码
var Bob = {
name: "Bob",
greet: function() {
alert( "Hi, my name is " + this.name );
}
};
var Bob = {
name: "Bob",
greet: function() {
alert( "Hi, my name is " + this.name );
}
};

在这里,我们创建了一个拥有两个成员的Object实例,一个简单的字符name,和一个包含对name引用的方法greet,当执行 Bob.greet()方法时,greet()函数用Bob作为this的对象。现在,让我们来看看同样的Bob.greet()在新的对象上执行会如何:
Js代码
var Alice = {
name: "Alice",
greet: Bob.greet
};
var Alice = {
name: "Alice",
greet: Bob.greet
};

考虑下面哪一个才是Alice.greet()的输出呢?
1. “Hi, my name is Bob”, or
2. “Hi, my name is Alice”
1. “Hi, my name is Bob”, or
2. “Hi, my name is Alice”
如果你选择的是“2”,对了!Bob的greet函数中“this.name”的直到函数真正执行时才会被解析,所以,当Alice.greet()被调用时,函数greet()的this被设置为了Alice。但是,如果调用函数本身,而不是作为某个对象的方法,会发生什么呢?看看下面这个:
Js代码
var unboundGreet = Bob.greet;
unboundGreet();
var unboundGreet = Bob.greet;
unboundGreet();

你预测函数unboundGreet()会输出什么呢?答案是“”Hi, my name is”。这是因为当一个未绑定的函数被调用时,“this”被设置为了全局对象window,而window.name并没有定义,所以就产生空字符。
下面的例子你将会看到隐含的未绑定会令调试很头痛:
Js代码

setTimeout( Bob.greet, 1000 ); // Call Bob.greet 1 second from now
setTimeout( Bob.greet, 1000 ); // Call Bob.greet 1 second from now 

开发者可能希望Bob.greet()会在一秒后调用,它绑定到了Bob上就会输出“Hi, my name is Bob”,但是这种预期并没有出现,相反,这个greet()方法只能称为未绑定,其输出结果同unboundGreet()是一样的。下面这个等效的代码或许能更清楚地说明其中的原由:
Js代码

var unboundGreet = Bob.greet;
setTimeout( unboundGreet, 1000 );
var unboundGreet = Bob.greet;
setTimeout( unboundGreet, 1000 ); 

在这里,很明显,在执行时间开始(一秒后)时,函数greet()是一个未绑定的。当你把一个对象的方法设为一个事件处理器时,相同情况也会发生。看看下面的HTML:
Html代码

<a id="clickme" href="">Greetings!</a>
<a id="clickme" href="">Greetings!</a>

添加如下的JavaScript:
Js代码

document.getElementById("clickme").onclick = Bob.greet;
document.getElementById("clickme").onclick = Bob.greet; 

可能我们预期这会触发链接输出“Hi, my name is Bob”,但当点击它时,结果又和调用unboundGreet()是一样的。
解决隐含未绑定的问题方法就是创建一个函数,用它来包装无范围的方法(the bound method)。大多数JavaScript库都包含了这样的助手,多数
时候是通过向Function.prototype添加一个bind方法来完成的。
这里就不用这种方式,下面用一个简单的例子说明如何建仓这样的包装器(wrapper)。
Js代码

var bobsGreet = function() {
  Bob.greet();
};

setTimeout( bobsGreet, 1000 );
var bobsGreet = function() {
  Bob.greet();
};
setTimeout( bobsGreet, 1000 );

或者更简单一点,
Js代码

setTimeout( function(){

  Bob.greet();

}, 1000 );
setTimeout( function(){
  Bob.greet();
}, 1000 ); 

这就是本文的全部内容了。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值