一、问题

学习 JavaScript 其中一个标志就是理解下面两种写法,会输出有不一样的结果。

var obj = {

  foo: function () {}

};

var foo = obj.foo;

// 写法一

obj.foo()

// 写法二

foo()

在上面的代码中,虽然 obj.foo 和 foo 指向同一个函数,但是执行结果不一样。

请看下面的示例。

var obj = {

  foo: function () { console.log(this.bar) },

  bar: 1

};

var foo = obj.foo;

var bar = 2;

obj.foo() //访问后的的结果为

foo() //访问后的结果为

形成这种差异的原因,就在于函数体内部使用了 this 关键字。在指导用书上, this 指的是函数运行时所在的环境。

★对于 obj.foo() 来说, foo 运行在 obj 环境,所以 this 指向 obj ;

 ★对于 foo() 来说, foo 运行在全局环境,所以 this 指向全局环境。

所以,两者的运行结果不一样。

上面这种解释没错,但是指导用书并不告诉你,为什么会这样?也就是说,函数的运行环境到底是怎么决定的?

举例来说,为什么 obj.foo() 就是在 obj 环境执行,而一旦 var foo = obj.foo , foo() 就变成在全局环境执行?

下面就来解释 JavaScript 这样处理的原理。理解了这一点,你就会彻底理解 this 的作用。

二、内存的数据结构

JavaScript 语言之所以有 this 的设计,跟内存里面的数据结构有关系。

var obj = { foo:  5 };

上面的代码将一个对象赋值给变量 obj 。JavaScript 引擎会先在内存里面,生成一个对象 { foo: 5 } ,然后把这个对象的内存地址赋值给变量 obj 。

  

也就是说,变量 obj 是一个地址(reference)。后面如果要读取 obj.foo ,引擎先从 obj 拿到内存地址,然后再从该地址读出原始的对象,返回它的 foo 属性。

原始的对象以字典结构保存,每一个属性名都对应一个属性描述对象。举例来说,上面例子的 foo 属性,实际上是以下面的形式保存的。

  

{
foo: {
[[value]]: 5
[[writable]]: true
[[enumerable]]: true
[[configurable]]: true
}
}

注意, foo 属性的值保存在属性描述对象的 value 属性里面。

三、函数

这样的结构是很清晰的,问题在于属性的值可能是一个函数。

var obj = { foo: function () {} };

这时,引擎会将函数单独保存在内存中,然后再将函数的地址赋值给 foo 属性的 value 属性。

  

{

  foo: {

    [[value]]: 函数的地址

    ...

  }

}

由于函数是一个单独的值,所以它可以在不同的环境(上下文)执行。

var f = function () {};

var obj = { f: f };

// 单独执行

f()

// obj 环境执行

obj.f()

四、环境变量

JavaScript 允许在函数体内部,引用当前环境的其他变量。

var f = function () {

  console.log(x);

};

上面代码中,函数体里面使用了变量 x 。该变量由运行环境提供。

现在问题就来了,由于函数可以在不同的运行环境执行,所以需要有一种机制,能够在函数体内部获得当前的运行环境(context)。所以, this 就出现了,它的设计目的就是在函数体内部,指代函数当前的运行环境。

var f = function () {

  console.log(this.x);

}

上面代码中,函数体里面的 this.x 就是指当前运行环境的 x 。

var f = function () {

  console.log(this.x);

}
var x = 1; var obj = { f: f, x: 2, }; // 单独执行
f() //
// obj 环境执行 obj.f() //

上面代码中,函数 f 在全局环境执行, this.x 指向全局环境的 this 。

  

在 obj 环境执行, this.x 指向 obj.x 。

  

现在回到文章开头提出的问题, obj.foo() 是通过 obj 找到 foo ,所以就是在 obj 环境执行。一旦 var foo = obj.foo ,变量 foo 就直接指向函数本身,所以 foo() 就变成在全局环境执行。

JavaScript 中 this 的原理的更多相关文章

  1. JavaScript中new实现原理

    JavaScript中new实现原理 1.创建一个空对象 obj 2.将该对象 obj 的原型链 __proto__ 指向构造函数的原型 prototype, 并且在原型链 __proto__ 上设置 ...

  2. JavaScript中的计时器原理

    理解John Resig 在 How JavaScript Timers Work. 原理分析 timer(setInterval,setTimeout)有一个很重要的概念,时间延迟的长短是不稳定的. ...

  3. javaScript中的闭包原理 (译)

    这篇文章通过javaScript代码解释了闭包的原理,来让编程人员理解闭包.它不是写给大牛或使用功能性语言进行编程的程序员的.一旦意会了其核心概念,闭包理解起来并不难.然而,你不可能通过阅读任何有关闭 ...

  4. 剖析Javascript中forEach()底层原理,如何重写forEach()

    我们平时用的forEach()一般是这样用的 var myArr = [1,5,8] myArr.forEach((v,i)=>{ console.log(v,i) })//运行后是这样的1 0 ...

  5. JavaScript中this对象原理简洁说明

    今天看了阮一峰大神的博客文章:JavaScript 的this原理,把纠结很久的this的指向终于理解清楚了 原文:http://www.ruanyifeng.com/blog/2018/06/jav ...

  6. JavaScript中的继承(原型链)

    一.原型链 ECMAScript中将原型链作为实现继承的主要方法,基本思想是利用原型让一个引用类型继承另一个引用类型的属性和方法. 实例1: function SupType() { this.pro ...

  7. 领悟 JavaScript 中的面向对象

    JavaScript是基于对象的语言,我们可以使用面向对象的思想去开发js代码. JavaScript是基于对象的语言. 可以使用面向对象的思想,但是不少人对这一点理解得并不全面. 在 JavaScr ...

  8. JavaScript中this的工作原理以及注意事项

    在JavaScript中,this 的概念比较复杂.除了在面向对象编程中,this 还是随处可用的.这篇文章介绍了this 的工作原理,它会造成什么样的问题以及this 的相关例子. 要根据this  ...

  9. 原生JavaScript中动画与特效的实现原理

    现如今,许多页面上均有一些动画效果.适当的动画效果可以在一定程度上提高页面的美观度,具有提示效果的动画可以增强页面的易用性. 实现页面动画的途径一般有两种. 一种是通过操作JavaScript间接操作 ...

随机推荐

  1. Repeat Number(数论)

    Repeat Number 题目描述: Definition: a+b = c, if all the digits of c are same ( c is more than ten), then ...

  2. Java Collections Framework知识结构目录

    The core collection interfaces are the foundation of the Java Collections Framework. The Java Collec ...

  3. 查询组成员(group)

    查询组成员 $groupname = "groupname" $members = (get-adgroup $groupname -properties member).memb ...

  4. c# 知识学习

    1.C#基础知识梳理系列 2.详解C#委托,事件与回调函数 3.C#制作Windows service

  5. yii2.0中url重写实现方法

    在yii框架里有前台和后台页面,举例前台url重写. 控制器与路由 控制器以Controller作为后缀,继承自yii\web\Controller; 动作以action作为前缀,public访问修饰 ...

  6. spring初始化完成后执行初始化数据方法

    Spring提供的解决方案三种: 1.InitializingBean package com.foriseland.fsoa.fabricca; import com.foriseland.fsoa ...

  7. January 14 2017 Week 2nd Saturday

    Don't try so hard, the best things come when you least expect them to. 不要着急,最好的总会在最不经意时出现. The secon ...

  8. OC NSMutableString的使用

    #pragma mark 可变字符串的创建 void stringCreate() { // 预先分配10个字数的存储空间 NSMutableString *str = [[NSMutableStri ...

  9. vim使用常看

    原网址http://www.runoob.com/linux/linux-vim.html 补充参考https://blog.csdn.net/w178191520/article/details/8 ...

  10. dom4j.jar有什么作用?

    om4j是一个Java的XML API,类似于jdom,用来读写XML文件的.dom4j是一个非常非常优秀的Java XML API,具有性能优异.功能强大和极端易用使用的特点,同时它也是一个开放源代 ...