PPK 谈 JavaScript 的 this 关键字 [翻译]


Posted in Javascript onSeptember 29, 2009

下面先讲如何在event handling(事件处理)中用它,再接着是讲 this 的其他用法。

自己本身

先来看看函数 doSomething() 里的 this 到底是指向(refer to)了什么?

function doSomething() {
  this.style.color = '#cc0000';
}

JavaScript的 this 总指向所运行的函数“自己本身”。也就是说,它是一种指向函数对象的方法。在页面中定义 doSomething() 函数,自己本身是指页面。也就是说,是指 JavaScript 的 window 对象(全局对象)。而 onclick 属性它自己本身是属 HTML 元素所有。

这个“所有权”是 JavaScript 的 OO(面向对象)特性的后果。在 把对象作关联数组 页面中有更多信息。

------------ window --------------------------------------
|                     / \      |
|                      |      |
|                     this     |
|  ----------------            |      |
|  | HTML element | <-- this     ----------------- |
|  ----------------   |      | doSomething() | |
|        |     |      ----------------- |
|     --------------------             |
|     | onclick property |             |
|     --------------------             |
|                            |
----------------------------------------------------------

如果 doSomething() 运行时没有任何与之预留相关的话,关键字 this 指向 window(窗口) ,该函数将会改动 window 的 style.color。而 window 没有 style 这样的对象,所以该函数会引发 JavaScript 的错误。

拷贝(copying)

因此,用好 this 有些难度。像在函数中使用的上面例子的这种情况,它应该指向 HTML 元素“自己本身”。换个说法是,有个函数拷贝指向 onclick 属性。 我们来看看在传统事件注册中的情况。

element.onclick = doSomething;

因为函数拷贝全指向了 onclick 属性(现在变成了方法),所以在事件处理执行时,this 指向 HTML 元素并将 color 改动。

------------ window --------------------------------------
|                            |
|                            |
|                            |
|  ----------------                   |
|  | HTML element | <-- this     ----------------- |
|  ----------------   |      | doSomething() | |
|        |     |      ----------------- |
|     -----------------------     |      |
|     |copy of doSomething()| <-- copy function  |
|     -----------------------            |
|                            |
----------------------------------------------------------

这可以让我们为多个事件处理给它函数拷贝。每次 this 将指向正确的 HTML 元素:

------------ window --------------------------------------
|                            |
|                            |
|                            |
|  ----------------                   |
|  | HTML element | <-- this     ----------------- |
|  ----------------   |      | doSomething() | |
|        |     |      ----------------- |
|     -----------------------     |      |
|     |copy of doSomething()| <-- copy function  |
|     -----------------------     |      |
|                      |      |
|  -----------------------         |      |
|  | another HTML element| <-- this    |      |
|  -----------------------   |      |      |
|        |        |      |      |
|     -----------------------     |      |
|     |copy of doSomething()| <-- copy function  |
|     -----------------------            |
|                            |
----------------------------------------------------------

每次函数被调用,this 指向当前所处理的事件的那个 HTML 元素(“自己本身” doSomething() 的拷贝)。

指向(referring)

要是用 行内事件注册呢?

<element onclick="doSomething()">

这里没有拷贝函数,而是指向它,有什么不一样呢? 这个 onclick 属性没有包含实际函数,而只是一个函数调用。

doSomething();

上面的意思是:“到 doSomething() 那里去执行它”。在doSomething()里面,this 关键字再次指向全局 window 对象,那么函数会返回错误的消息。

------------ window --------------------------------------
|                     / \      |
|                      |      |
|                     this     |
|  ----------------            |      |
|  | HTML element | <-- this     ----------------- |
|  ----------------   |      | doSomething() | |
|        |     |      ----------------- |
|     -----------------------     / \      |
|     | go to doSomething() |     |      |
|     | and execute it   | ---- reference to   |
|     -----------------------    function    |
|                            |
----------------------------------------------------------

不一样?

如果是用 this 去访问 HTML 元素来处理事件的话,那么必须肯定它实际是写入了 onclick 属性中。而它指向 HTML 元素的事件处理就算已注册。如果这么做:

element.onclick = doSomething;
alert(element.onclick)

得到的是

function doSomething()
{
	this.style.color = '#cc0000';
}

可以看到,this 关键字在 onclick 方法中。它指向 HTML 元素。

但是如果这么做:

<element onclick="doSomething()">
alert(element.onclick)

得到的是

function onclick()
{
	doSomething()
}

这里只是指向函数 doSomething()。this 关键字不在 onclick 方法中。它没有指向 HTML 元素。

例子-拷贝

在下面示例中,this 写入 onclick 方法中:

element.onclick = function () {doSomething()}
element.attachEvent('onclick',doSomething)
<element onclick="doSomething()">

例子-指向

在下面示例中,this 指向 window:

element.onclick = function () {doSomething()}
element.attachEvent('onclick',doSomething)
<element onclick="doSomething()">

要注意上面的 attachEvent。它的缺点是微软事件注册模型,它创建了指向该函数,而且没有拷贝它。所以有时不可能弄清楚 HTML 当前的处理事件是哪个。

结合

使用行内事件注册时,也可以把 this 发送到函数。所以可以这么用:

<element onclick="doSomething(this)">
 
function doSomething(obj) {
	// this is present in the event handler and is sent to the function
	// obj now refers to the HTML element, so we can do
	obj.style.color = '#cc0000';
}
Javascript 相关文章推荐
使用Javascript接收get传递的值的代码
Nov 30 Javascript
Javascript写入txt和读取txt文件示例
Feb 12 Javascript
JS实现的4种数字千位符格式化方法分享
Mar 02 Javascript
javascript伸缩菜单栏实现代码分享
Nov 12 Javascript
jQuery实现最简单的切换图效果【可兼容IE6、火狐、谷歌、opera等】
Sep 04 Javascript
移动适配的几种方案(三种方案)
Nov 25 Javascript
jQuery实现对象转为url参数的方法
Jan 11 Javascript
10道典型的JavaScript面试题
Mar 22 Javascript
使用JS和canvas实现gif动图的停止和播放代码
Sep 01 Javascript
js如何找出字符串中的最长回文串
Jun 04 Javascript
axios对请求各种异常情况处理的封装方法
Sep 25 Javascript
详解如何修改 node_modules 里的文件
May 22 Javascript
一个JS小玩意 几个属性相加不能超过一个特定值.
Sep 29 #Javascript
IE FF OPERA都可用的弹出层实现代码
Sep 29 #Javascript
javascript 表单验证常见正则
Sep 28 #Javascript
javascript 页面划词搜索JS
Sep 28 #Javascript
jquery 模式对话框终极版实现代码
Sep 28 #Javascript
javascript的onchange事件与jQuery的change()方法比较
Sep 28 #Javascript
支持ie与FireFox的剪切板操作代码
Sep 28 #Javascript
You might like
一台收音机,让一家人都笑逐颜开!
2020/08/21 无线电
PHP与javascript对多项选择的处理
2006/10/09 PHP
PHP中uploaded_files函数使用方法详解
2011/03/09 PHP
解析PHP实现多进程并行执行脚本
2013/06/18 PHP
数组方法解决JS字符串连接性能问题有争议
2011/01/12 Javascript
js中widow.open()方法使用详解
2013/07/30 Javascript
jQuery中html()方法用法实例
2014/12/25 Javascript
DOM基础教程之模型中的模型节点
2015/01/19 Javascript
基于jquery实现简单的手风琴特效
2015/11/24 Javascript
jquery实现鼠标悬浮停止轮播特效
2020/08/20 Javascript
基于jQuery日历插件制作日历
2016/03/11 Javascript
Google 爬虫如何抓取 JavaScript 的内容
2017/04/07 Javascript
基于jstree使用AJAX请求获取数据形成树
2017/08/29 Javascript
nodejs async异步常用函数总结(推荐)
2017/11/17 NodeJs
JS/jQuery实现DIV延时几秒后消失或显示的方法
2018/02/12 jQuery
小程序卡片切换效果组件wxCardSwiper的实现
2020/02/13 Javascript
ES6中的类(Class)示例详解
2020/12/09 Javascript
在Django的URLconf中使用命名组的方法
2015/07/18 Python
python删除文本中行数标签的方法
2018/05/31 Python
Python实现监控键盘鼠标操作示例【基于pyHook与pythoncom模块】
2018/09/04 Python
Python使用gRPC传输协议教程
2018/10/16 Python
用python写一个定时提醒程序的实现代码
2019/07/22 Python
用Python实现二叉树、二叉树非递归遍历及绘制的例子
2019/08/09 Python
Python3.7 读取 mp3 音频文件生成波形图效果
2019/11/05 Python
Python虚拟环境venv用法详解
2020/05/25 Python
利用django创建一个简易的博客网站的示例
2020/09/29 Python
使用HTML5 Geolocation实现一个距离追踪器
2018/04/09 HTML / CSS
海外淘书首选:AbeBooks
2017/07/31 全球购物
全球独特生活方式产品和礼品购物网站:AHAlife
2018/09/18 全球购物
Set里的元素是不能重复的,那么用什么方法来区分重复与否呢? 是用==还是equals()? 它们有何区别?
2014/07/27 面试题
2014年开学第一课活动方案
2014/03/06 职场文书
村党支部书记四风问题个人对照检查材料思想汇报
2014/10/06 职场文书
承诺函范文
2015/01/21 职场文书
2015年健康教育工作总结
2015/04/10 职场文书
乱世佳人观后感
2015/06/08 职场文书
小程序实现筛子抽奖
2021/05/26 Javascript