浅谈JS的作用域链(三)
前面两篇文章介绍了JavaScript执行上下文中两个重要属性:VO/AO和scope chain。本文就来看看执行上下文中的this。
首先看看下面两个对this的概括:
- this是执行上下文(Execution Context)的一个重要属性,是一个与执行上下文相关的特殊对象。因此,它可以叫作上下文对象(也就是用来指明执行上下文是在哪个上下文中被触发的对象)。
- this不是变量对象(Variable Object)的一个属性,所以跟变量不同,this从不会参与到标识符解析过程。也就是说,在代码中当访问this的时候,它的值是直接从执行上下文中获取的,并不需要任何作用域链查找。this的值只在进入上下文的时候进行一次确定。
关于this最困惑的应该是,同一个函数,当在不同的上下文进行调用的时候,this的值就可能会不同。也就是说,this的值是通过函数调用表达式(也就是函数被调用的方式)的caller所提供的。
下面就看看在不同场景中,this的值。
全局上下文
在全局上下文(Global Context)中,this总是global object,在浏览器中就是window对象。
1
2
3
4
5
6
7
8
9
10
|
console.log( this === window); this .name = "Will" ; this .age = 28 ; this .getInfo = function(){ console.log( this .name + " is " + this .age + " years old" ); }; window.getInfo(); // true // Will is 28 years old |
函数上下文
在一个函数中,this的情况就比较多了,this的值直接受函数调用方式的影响。
Invoke function as Function
当通过正常的方式调用一个函数的时候,this的值就会被设置为global object(浏览器中的window对象)。
但是,当使用"strict mode"执行下面代码的时候,this就会被设置为"undefined"。
1
2
3
4
5
6
7
8
|
function gFunc(){ return this ; } console.log(gFunc()); console.log( this === window.gFunc()); // window // true |
Invoke function as Method
当函数作为一个对象方法来执行的时候,this的值就是该方法所属的对象。
在下面的例子中,创建了一个obj对象,并设置name属性的值为"obj"。所以但调用该对象的func方法的时候,方法中的this就表示obj这个对象。
1
2
3
4
5
6
7
8
9
|
var obj = { name: "obj" , func: function () { console.log( this + ":" + this .name); } }; obj.func(); // [object Object]:obj |
为了验证"方法中的this代表方法所属的对象"这句话,再看下面一个例子。
在对象obj中,创建了一个内嵌对象nestedObj,当调用内嵌对象的方法的时候,方法中的this就代表nestedObj。
1
2
3
4
5
6
7
8
9
10
11
12
|
var obj = { name: "obj" , nestedObj: { name: "nestedObj" , func: function () { console.log( this + ":" + this .name); } } } obj.nestedObj.func(); // [object Object]:nestedObj |
对于上面例子中的方法,通常称为绑定方法,也就是说这些方法都是个特定的对象关联的。
但是,当我们进行下面操作的时候,temp将是一个全局作用里面的函数,并没有绑定到obj对象上。所以,temp中的this表示的是window对象。
1
2
3
4
5
6
7
8
9
10
11
|
var name = "Will" ; var obj = { name: "obj" , func: function () { console.log( this + ":" + this .name); } }; temp = obj.func; temp(); // [object Window]:Will |
Invoke function as Constructor
在JavaScript中,函数可以作为构造器来直接创建对象,在这种情况下,this就代表了新建的对象。
1
2
3
4
5
6
7
8
9
10
11
|
function Staff(name, age){ this .name = name; this .age = age; this .getInfo = function(){ console.log( this .name + " is " + this .age + " years old" ); }; } var will = new Staff( "Will" , 28 ); will.getInfo(); // Will is 28 years old |
Invoke context-less function
对于有些没有上下文的函数,也就是说这些函数没有绑定到特定的对象上,那么这些上下文无关的函数将会被默认的绑定到global object上。
在这个例子中,函数f和匿名函数表达式在被调用的过程中并没有被关联到任何对象,所以他们的this都代表global object。
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 context = "global" ; var obj = { context: "object" , method: function () { console.log( this + ":" + this .context); function f() { var context = "function" ; console.log( this + ":" + this .context); }; f(); (function(){ var context = "function" ; console.log( this + ":" + this .context); })(); } }; obj.method(); // [object Object]:object // [object Window]:global // [object Window]:global |
call/apply/bind改变this
this本身是不可变的,但是 JavaScript中提供了call/apply/bind三个函数来在函数调用时设置this的值。
这三个函数的原型如下:
fun.apply(obj1 [, argsArray])
- Sets obj1 as the value of this inside fun() and calls fun() passing elements of argsArray as its arguments.
fun.call(obj1 [, arg1 [, arg2 [,arg3 [, ...]]]])
- Sets obj1 as the value of this inside fun() and calls fun() passing arg1, arg2, arg3, ... as its arguments.
fun.bind(obj1 [, arg1 [, arg2 [,arg3 [, ...]]]])
- Returns the reference to the function fun with this inside fun() bound to obj1 and parameters of fun bound to the parameters specified arg1, arg2, arg3, ....
下面看一个简单的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
function add(numA, numB){ console.log( this .original + numA + numB); } add( 1 , 2 ); var obj = {original: 10 }; add.apply(obj, [ 1 , 2 ]); add.call(obj, 1 , 2 ); var f1 = add.bind(obj); f1( 2 , 3 ); var f2 = add.bind(obj, 2 ); f2( 3 ); // NaN // 13 // 13 // 15 // 15 |
当直接调用add函数的时候,this将代表window,当执行"this.original"的时候,由于window对象并没有"original"属性,所以会得到"undefined"。
通过call/apply/bind,达到的效果就是把add函数绑定到了obj对象上,当调用add的时候,this就代表了obj这个对象。
DOM event handler
当一个函数被当作event handler的时候,this会被设置为触发事件的页面元素(element)。
1
2
3
4
5
|
var body = document.getElementsByTagName( "body" )[ 0 ]; body.addEventListener( "click" , function(){ console.log( this ); }); // <body>…</body> |
In-line event handler
当代码通过in-line handler执行的时候,this同样指向拥有该handler的页面元素。
看下面的代码:
1
2
3
4
|
document.write( '<button onclick="console.log(this)">Show this</button>' ); // <button onclick="console.log(this)">Show this</button> document.write( '<button onclick="(function(){console.log(this);})()">Show this</button>' ); // window |
在第一行代码中,正如上面in-line handler所描述的,this将指向"button"这个element。但是,对于第二行代码中的匿名函数,是一个上下文无关(context-less)的函数,所以this会被默认的设置为window。
前面我们已经介绍过了bind函数,所以,通过下面的修改就能改变上面例子中第二行代码的行为:
document.write('<button onclick="((function(){console.log(this);}).bind(this))()">Show this</button>'); // <button onclick="((function(){console.log(this);}).bind(this))()">Show this</button>
保存this
在JavaScript代码中,同一个上下文中可能会出现多个this,为了使用外层的this,就需要对this进行暂存了。
看下面的例子,根据前面的介绍,在body元素的click handler中,this肯定是指向body这个元素,所以为了使用"greeting"这个方法,就是要对指向bar对象的this进行暂存,这里用了一个self变量。
有了self,我们就可以在click handler中使用bar对象的"greeting"方法了。
当阅读一些JavaScript库代码的时候,如果遇到类似self,me,that的时候,他们可能就是对this的暂存。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
var bar = { name: "bar" , body: document.getElementsByTagName( "body" )[ 0 ], greeting: function(){ console.log( "Hi there, I'm " + this + ":" + this .name); }, anotherMethod: function () { var self = this ; this .body.addEventListener( "click" , function(){ self.greeting(); }); } }; bar.anotherMethod(); // Hi there, I'm [object Object]:bar |
同样,对于上面的例子,也可以使用bind来设置this达到相同的效果。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
var bar = { name: "bar" , body: document.getElementsByTagName( "body" )[ 0 ], greeting: function(){ console.log( "Hi there, I'm " + this + ":" + this .name); }, anotherMethod: function () { this .body.addEventListener( "click" , (function(){ this .greeting(); }).bind( this )); } }; bar.anotherMethod(); // Hi there, I'm [object Object]:bar |
总结
本文介绍了执行上下文中的this属性,this的值直接影响着代码的运行结果。
在函数调用中,this是由激活上下文代码的调用者(caller)来提供的,即调用函数的父上下文(parent context ),也就是说this取决于调用函数的方式,指向调用时所在函数所绑定的对象。
浅谈JS的作用域链(三)的更多相关文章
- 浅谈JS的作用域链(一)
JS的执行环境 执行环境(Execution context,EC)或执行上下文,是JS中一个极为重要的概念. 在JavaScript中有三种代码运行环境: Global Code JavaScrip ...
- 浅谈JS的作用域链(二)
上一篇文章中介绍了Execution Context中的三个重要部分:VO/AO,scope chain和this,并详细的介绍了VO/AO在JavaScript代码执行中的表现. 本文就看看Exec ...
- 浅谈 js eval作用域
原文:浅谈 js eval作用域 就简单聊下如何全局 eval 一个代码. var x = 1; (function () { eval('var x = 123;'); })(); console. ...
- 浅谈js变量作用域
变量的作用域也是前端面试题常考的一个问题,掌握下面几个规律可以帮你更好的理解js的作用域. 1.作用域优先级遵循就近原则,函数内部的作用域优先级大于外部 var a=456; var b=111; f ...
- 浅谈JS中的闭包
浅谈JS中的闭包 在介绍闭包之前,我先介绍点JS的基础知识,下面的基础知识会充分的帮助你理解闭包.那么接下来先看下变量的作用域. 变量的作用域 变量共有两种,一种为全局变量,一种为局部变量.那么全局变 ...
- 浅谈JS之AJAX
0x00:什么是Ajax? Ajax是Asynchronous Javascript And Xml 的缩写(异步javascript及xml),Ajax是使用javascript在浏览器后台操作HT ...
- JS 之作用域链和闭包
1.JS无块级作用域 <script> function Main(){ if (1==1){ var name = "alex"; } console.log(nam ...
- 浅谈 js 语句块与标签
原文:浅谈 js 语句块与标签 语句块是什么?其实就是用 {} 包裹的一些js代码而已,当然语句块不能独立作用域.可以详细参见这里<MDN block> 也许很多人第一印象 {} 不是对象 ...
- 浅谈JS面向对象
浅谈JS面向对象 一 .什么是面向过程 就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以了.注重代码的过程部分. 二.什么是面向对象 最先出现在管理学 ...
随机推荐
- C#语言————第四章 深入C#的String类
*********类型转换**************** Convert与Parse的区别: Convert可以将任何内置类型转换为其他任何内置类型 XX.Parse:只能将字符串转换为XX类型例如 ...
- 1. svg学习笔记-在网页中使用svg
在网页中使用svg有以下三种方式 1. svg归根结底来说是一种图像格式,虽然有别于jpeg,gif,png等位图图像格式,所以在网页中能嵌入图像的地方都可以嵌入svg,例如将svg文件设置为< ...
- js入门-文本框输入特定内容控制另一个文本框
在填写表单时,有时需要某些文本框隐藏,当一文本框输入特定内容时才会显示隐藏的文本框,这一功能可以用onchange事件或oninput事件实现.下面对比下两种方法实现的区别: onchange()定义 ...
- zabbix疑难之时区问题
zabbix疑难总结: 1.zabbix的web界面的时间不对.晚12个小时整 适用于:安装网上的说法来修改,但是时间仍然不对的情况 我们按照以前的网上的那些方法在配置zabbix,需要配置时区: ...
- 查看tomcat项目中,具体占用cpu高的线程。
1.查看主进程占用cpu高: 此处主进程:27823 ~]# top top - :0: up days, :, 3 users, load average: 13.12, 13.31, 13.23 ...
- gcc库链接
转载于https://blog.csdn.net/zhangdaisylove/article/details/45721667 1.库的分类 库有静态库和动态库,linux下静态库为.a,动态库为. ...
- c++11の的左值、右值以及move,foward
左值和右值的定义 在C++中,可以放到赋值操作符=左边的是左值,可以放到赋值操作符右边的是右值.有些变量既可以当左值又可以当右值.进一步来讲,左值为Lvalue,其实L代表Location,表示在内存 ...
- Django之views
一 URL补充 二 Views试图函数 一 URL补充 1 MTV模型 2 django建立流程(用命令版) (1)django-admin startproject projectname (2) ...
- MongoDB逻辑操作符$or, $and,$not,$nor
$or是一个逻辑or操作符操作在一个数据或者多个表达式并且需要选择至少一个满足条件的表达式,$or有至少以下表达式: { $or: [ { <expression1> }, { <e ...
- ubuntu 16.04 SS安装及配置
安装SS客户端 安装pip3 一般情况下,pip3安装的版本比pip安装的新,pip安装的版本比apt安装的新,这里选择最新版本. sudo apt install python3-pip 安装SS ...