前面两篇文章介绍了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(12);
 
var obj = {original: 10};
add.apply(obj, [12]);
add.call(obj, 12);
 
var f1 = add.bind(obj);
f1(23);
 
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的作用域链(三)的更多相关文章

  1. 浅谈JS的作用域链(一)

    JS的执行环境 执行环境(Execution context,EC)或执行上下文,是JS中一个极为重要的概念. 在JavaScript中有三种代码运行环境: Global Code JavaScrip ...

  2. 浅谈JS的作用域链(二)

    上一篇文章中介绍了Execution Context中的三个重要部分:VO/AO,scope chain和this,并详细的介绍了VO/AO在JavaScript代码执行中的表现. 本文就看看Exec ...

  3. 浅谈 js eval作用域

    原文:浅谈 js eval作用域 就简单聊下如何全局 eval 一个代码. var x = 1; (function () { eval('var x = 123;'); })(); console. ...

  4. 浅谈js变量作用域

    变量的作用域也是前端面试题常考的一个问题,掌握下面几个规律可以帮你更好的理解js的作用域. 1.作用域优先级遵循就近原则,函数内部的作用域优先级大于外部 var a=456; var b=111; f ...

  5. 浅谈JS中的闭包

    浅谈JS中的闭包 在介绍闭包之前,我先介绍点JS的基础知识,下面的基础知识会充分的帮助你理解闭包.那么接下来先看下变量的作用域. 变量的作用域 变量共有两种,一种为全局变量,一种为局部变量.那么全局变 ...

  6. 浅谈JS之AJAX

    0x00:什么是Ajax? Ajax是Asynchronous Javascript And Xml 的缩写(异步javascript及xml),Ajax是使用javascript在浏览器后台操作HT ...

  7. JS 之作用域链和闭包

    1.JS无块级作用域 <script> function Main(){ if (1==1){ var name = "alex"; } console.log(nam ...

  8. 浅谈 js 语句块与标签

    原文:浅谈 js 语句块与标签 语句块是什么?其实就是用 {} 包裹的一些js代码而已,当然语句块不能独立作用域.可以详细参见这里<MDN block> 也许很多人第一印象 {} 不是对象 ...

  9. 浅谈JS面向对象

    浅谈JS面向对象 一 .什么是面向过程 就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以了.注重代码的过程部分. 二.什么是面向对象 最先出现在管理学 ...

随机推荐

  1. SDE ST_Geometry SQL st_intersects查询很慢的解决方法

    环境:服务端 SDE 10.0 oracle 11.2,客户端 PLSQL 11,oracle 11.2 为了调试方便,以下测试都是把sql提取出来在PLSQL上做 需求是已知一个多边形的点坐标,要在 ...

  2. python 在windows下的 虚拟环境

    解决 python 环境问题 windows 下安装 pip install virtualenv virtualenv的基本使用 1.1 创建虚拟环境 virtualenv venv 为环境指定Py ...

  3. C++整形转化成string类型---路径拼接在批处理程序中的应用

    上"酸菜" // show_dateset_image.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include ...

  4. 【转】避免全表扫描的sql优化

    对查询进行优化,应尽量避免全表扫描,首先应考虑在where 及order by 涉及的列上建立索引: .尝试下面的技巧以避免优化器错选了表扫描:· 使用ANALYZE TABLE tbl_name为扫 ...

  5. AirSim

    https://github.com/Microsoft/AirSim 功能 1 虚拟模拟 2半虚拟模拟 安装教程 环境安装 1安装 cmake 直接下 .exe 2安装cuda 3安装Eigen 3 ...

  6. JS进阶之---执行上下文,变量对象,变量提升

    一.结构顺序大体介绍 JavaScript代码的整个执行过程,分为两个阶段,代码编译阶段与代码执行阶段. 编译阶段由编译器完成,将代码翻译成可执行代码,这个阶段作用域规则会确定. 执行阶段由引擎完成, ...

  7. mascara-1

    来源:https://github.com/MetaMask/mascara (beta) Add MetaMask to your dapp even if the user doesn't hav ...

  8. win10下乌龟git安装和使用(转)

    文章转自http://blog.csdn.net/jdsjlzx/article/details/51098588 一.安装git for windows 首先下载Git for windows客户端 ...

  9. css学习之样式层级和权重

    第一种情况 当选择器相同的情况下,引入方式的前后,决定页面最后的效果 ---------外部在最后面显示 ---------内部在最后面显示 第二种情况 引入方式相同时候,则是按照权重取最大(取权重最 ...

  10. navicate使用小技巧

    以下效果是按住alt+按住鼠标左键,往左往右移动,即可