JavaScript中的this关键字

“this”关键字是JavaScript中广泛应用的一种特性,但它经常也是这门语言中最容易混淆和误解的特性。“this”的实际意义是什么?它是如何求值的?

本文试图以清晰的方式澄清和解释这问题的答案。

有过其他编程经验的人对“this”关键字并不陌生,大部分时候当通过构造函数实例化一个类的实例时,它指新创建的对象。例如,如果我有一个类Boat(),其拥有一个moveBoat()方法,当在moveBoat方法中引用“this”的时候,我们实际上访问的基于Boat类新创建的对象。

在JavaScript中,当通过“new”关键字调用构造函数时,我们也有this概念,然而,这并不是唯一的规则,并且“this”经常在不同的执行上下文中引用到不同的对象。如果你不熟悉JavaScript中的执行上下文,我建议你阅读我的另一篇文章。说的够多了,让我们看一些JavaScript的例子:

// 全局作用域

foo = ‘abc‘;
alert(foo); // abc

this.foo = ‘def‘;
alert(foo); // def

每当你在全局作用域中使用“this”关键字时(没在函数内部),它通常指向全局对象(global
object)。现在让我们看看函数内部“this”的值:

var boat = {
size: ‘normal‘,
boatInfo: function() {
alert(this === boat);
alert(this.size);
}
};

boat.boatInfo(); // true, ‘normal‘

var bigBoat = {
size: ‘big‘
};

bigBoat.boatInfo = boat.boatInfo;
bigBoat.boatInfo(); // false, ‘big‘

那么上面的“this”如何确定?我们看到上面的boat对象有一个size属性和一个boatInfo方法。在boatInfo()内部,会弹出this的值是否是boat对象,也会弹出this的size属性。所以我们执行boat.boatInfo(),我们看见this的值是boat对象和boat的size属性值是normal。

然后我们创建另一个对象bigBoat,也有一个size属性是big。然而,bigBoat对象没有boatInfo方法,所以我们从boat对象拷贝方法
bigBoat.boatInfo =
boat.boatInfo。现在,当我们调用bigBoat.boatInfo()并进入函数时,我们看到this不等于boat,并且现在size属性值是big。为什么会这样?boatInfo()内部的this值是如何改变的?

你必须意识到的第一件事是函数内部this的值不是静态的,每次你调用一个函数它总是重新求值,但这一过程发生在函数代码实际执行之前。函数内部的this值实际由函数被调用的父作用域提供,更重要的是,依赖实际函数的语法。

当函数被调用时,我们看紧邻括号“()”的左边。如果在括号的左侧存在一个引用,传递给调用函数的“this”值是引用属于的对象,否则this的值将是全局对象。让我们看一个例子:

function bar() {
alert(this);
}
bar(); // global - 因为bar方法被调用时属于 global 对象

var foo = {
baz: function() {
alert(this);
}
}
foo.baz(); // foo - 因为baz()方法被调用时术语foo对象

如果this就这么简单,那上面的代码就足够了。我们可以进一步使事情变得复杂,通过不同的调用语法,改变相同函数内部“this”的值。

var foo = {
baz: function() {
alert(this);
}
}
foo.baz(); // foo - 因为baz被调用时属于foo对象

var anotherBaz = foo.baz;
anotherBaz(); // global - 因为anotherBaz()被调用时术语global对象

我们看到baz()内部的“this”值每次都不同,这是因为调用的语法不同。现在,让我们看看深度嵌套对象内部“this”的值:

var anum = 0;

var foo = {
anum: 10,
baz: {
anum: 20,
bar: function() {
console.log(this.anum);
}
}
}
foo.baz.bar(); // 20 - 因为()的左边是bar,而它被调用时属于baz对象

var hello = foo.baz.bar;
hello(); // 0 - 因为()的左边是hello,而它被调用时属于global对象

另一个经常被问的问题是事件处理程序内部的“this”关键字如何求值?答案是事件处理程序内部的“this”总是引用触发事件的元素。让我们看一个例子:

<div id="test">I am an element with id #test</div>

function doAlert() {
alert(this.innerHTML);
}

doAlert(); // undefined

var myElem = document.getElementById(‘test‘);
myElem.onclick = doAlert;

alert(myElem.onclick === doAlert); // true
myElem.onclick(); // I am an element

我们看到当doAlert()第一次调用时,弹出的值是undefined,由于doAlert()属于global对象。然后我们写myElem.onclick
= doAlert。这意味这当onclick被出发时,它作为myElem的一个方法,“this”的值将是myElem元素。

我想说的最后一点是,“this”的值也可以通过call和apply手动设置,这超过我们所讨论的范围。还感兴趣的是,当调用构造函数时,“this”引用新创建的实例对象。原因是因为构造函数前面的“new”关键字,它创建一个新对象,构造函数内部的“this”总引用新创建的对象。

总结

希望今天的文章已经澄清了“this”关键字的误解,并且你总能知道“this”的正确值。现在我们知道“this”的值不是静态的,值得确定依赖于函数被如何调用。

原文 http://davidshariff.com/blog/javascript-this-keyword/

相关阅读



Q群推荐

CSS家园 188275051,Web开发者(前后端)的天堂,欢迎有兴趣的同学加入

GitHub家园 225932282,Git/GitHub爱好者的天堂,欢迎有兴趣的同学加入

码农之家 203145707,码农们的天堂,欢迎有兴趣的同学加入

JavaScript中的this关键字,码迷,mamicode.com

时间: 2024-10-26 04:19:49

JavaScript中的this关键字的相关文章

转载 深入理解JavaScript中的this关键字

转载原地址: http://www.cnblogs.com/rainman/archive/2009/05/03/1448392.html 深入理解JavaScript中的this关键字 1. 一般用处 2. this.x 与 apply().call() 3. 无意义(诡异)的this用处 4. 事件监听函数中的this 5. 总结 在JavaScript中this变量是一个令人难以摸清的关键字,this可谓是非常强大,充分了解this的相关知识有助于我们在编写面向对象的JavaScript程

JavaScript中的this关键字(copy别人的)

颜海镜 专注web前端,分享html,css,javascript,jquery等相关原创知识-- JavaScript中的this关键字 "this"关键字是JavaScript中广泛应用的一种特性,但它经常也是这门语言中最容易混淆和误解的特性."this"的实际意义是什么?它是如何求值的? 本文试图以清晰的方式澄清和解释这问题的答案. 有过其他编程经验的人对"this"关键字并不陌生,大部分时候当通过构造函数实例化一个类的实例时,它指新创建的

JavaScript中的let关键字

JavaScript中的let关键字 JavaScript中使用var定义变量可能存在的问题 var定义的变量没有块作用域 var定义的全局变量会自动添加全局window对象的属性 var定义的变量会提前装载 而let关键字的出现解决了以上的问题: <script type="text/javascript">                                for( let i = 0;i<10;i++){                      

关于Javascript中通过var关键字声明变量和function关键字声明函数的笔记

一.概念 1.变量声明 在JavaScript中,变量一般通过var关键字(隐式声明,let关键字声明除外)进行声明,如下通过var关键字声明a,b,c三个变量(并给其中的a赋值): var a=1,b,c; //关键字显式声明变量a,b,c,并给a赋值console.log(a); //1 //由于b,c未定义变量类型,因此输出"undefined"console.log(b); //undefinedconsole.log(c); //undefined //如果变量未声明,则输出

javascript 中的 this 关键字详解

1.javascript 中 什么是 this? this 指的是当前行为执行的主体,或者是当前方法执行的主体 context:是当前行为或者方法执行的环境 实例: xx 去北京饭店吃东西:上下文是“北京饭店”, this 是 xx 2. 那么如何判断一个函数在执行的时候,函数体内的 this 关键字是谁呢?主要有以下几条规律:  1)一个函数体内的 this 关键字和这个函数的在哪里定义,哪里执行都没有关系;   2) 判断一个方法执行的时候,函数的执行主体是谁?主要看方法前面有没有点(.),

深入理解JavaScript中的this关键字

1. 一般用处 2. this.x 与 apply().call() 3. 无意义(诡异)的this用处 4. 事件监听函数中的this 5. 总结 在JavaScript中this变量是一个令人难以摸清的关键字,this可谓是非常强大,充分了解this的相关知识有助于我们在编写面向对象的JavaScript程序时能够游刃有余. 1. 一般用处 对于this变量最要的是能够理清this所引用的对象到底是哪一个,也许很多资料上都有自己的解释,但有些概念讲的偏繁杂.而我的理解是:首先分析this所在

用自然语言的角度理解JavaScript中的this关键字

转自:http://blog.leapoahead.com/2015/08/31/understanding-js-this-keyword/ 在编写JavaScript应用的时候,我们经常会使用this关键字.那么this关键字究竟是怎样工作的?它的设计有哪些好的地方,有哪些不好的地方?本文带大家全面系统地认识这个老朋友. 这里的小明是主语,如果没有这个主语,那么后面的代词『他』将毫无意义.有了主语,代词才有了可以指代的事物. 类比到JavaScript的世界中,我们在调用一个对象的方法的时候

正确理解JavaScript中的this关键字

JavaScript有this关键字,this跟JavaScript的执行上下文密切相关,很多前端开发工程师至今对this关键字还是模棱两可,本文将结合代码讲解下JavaScript的this关键字. this和对象的关系 首先来看下面的代码: var person = { name:'Theo Wong', gender:'male', getName:function(){ console.log(person.name); } }; person.getName(); 定义了一个perso

深入理解javascript中的this 关键字(转载)

在JavaScript中this变量是一个令人难以摸清的关键字,this可谓是非常强大,充分了解this的相关知识有助于我们在编写面向对象的JavaScript程序时能够游刃有余. 1.一般用处 对于this关键字最重要的是明确this所引用的对象是哪一个,也许很多资料上都有自己的解释,但有些概念将的偏繁杂.而我的理解是:首先分析this所在的函数是当做哪个对象的方法调用的,则该对象就是this所引用的对象. Example 一: var obj = {}; obj.x = 100; obj.y