资讯专栏INFORMATION COLUMN

全方位解读this

yexiaobai / 2482人阅读

摘要:的构造函数等同于下。创建一个新的对象将构造函数的指向这个新对象指向构造函数的代码,为这个对象添加属性,方法等返回新对象。

原文链接 - http://www.jianshu.com/p/d647aa6d1ae6

首先,了解下执行上下文的生命周期。

在执行上下文的创建阶段,会分别生成变量对象,建立作用域链,以及确定this指向。
其中this的指向,是在函数被调用的时候确定的。也就是执行上下文被创建时确定的。
所以,同一个函数由于调用方式的不同,this指向了不一样的对象。

var a = 10;
var obj = {
    a: 20
}

function fn () {
    console.log(this.a);
}

fn(); // 10
fn.call(obj); // 20

另一个要注意的地方是,在函数执行过程中,this一旦被确定,就不可更改了

var a = 10;
var obj = {
    a: 20
}

function fn () {
    this = obj; // 这句话试图修改this,运行后会报错
    console.log(this.a);
}

fn();
一、全局对象中的this

关于全局对象的this,是一个比较特殊的存在。全局环境中的this,指向window本身。因此,这也相对简单,没有那么多复杂的情况需要考虑。

// 通过this绑定到全局对象
this.a2 = 20;

// 通过声明绑定到变量对象,但在全局环境中,变量对象就是它自身
var a1 = 10;

// 仅仅只有赋值操作,标识符会隐式绑定到全局对象
a3 = 30;

// 输出结果会全部符合预期
console.log(a1, a2, a3);
console.log(window.a1, window.a2, window.a3);
二、函数中的this

在总结函数中this指向之前,我想我们有必要通过一些奇怪的例子,来感受一下函数中this的捉摸不定。

// demo01
var a = 20;
function fn() {
    console.log(this.a);
}
fn();
// demo02
var a = 20;
function fn() {
    function foo() {
        console.log(this.a);
    }
    foo();
}
fn();
// demo03
var a = 20;
var obj = {
    a: 10,
    c: this.a + 20,
    fn: function () {
        return this.a;
    }
}

console.log(obj.c);
console.log(obj.fn());

在一个函数上下文中,this由调用者提供,由调用函数的方式来决定。
如果调用者函数,被某一个对象所拥有,那么该函数在调用时,内部的this指向该对象。
在严格模式下,如果函数独立调用,那么该函数内部的this,则指向undefined。
在非严格模式下,当this指向undefined时,它会被自动指向全局对象。

从结论中我们可以看出,想要准确确定this指向,找到函数的调用者以及区分他是否是独立调用就变得十分关键。

// 为了能够准确判断,我们在这里使用严格模式,因为非严格模式会自动指向全局
"use strict";
function fn() {
    console.log(this);
}

fn();  // fn是调用者,独立调用,undefined
window.fn();  // fn是调用者,被window所拥有,window

但是需要特别注意的是,在上面的demo03中,对象obj中的c属性使用this.a + 20来计算,而他的调用者obj.c并非是一个函数。因此不适用于上面的规则,我们要对这种方式多带带下一个结论。

当obj在全局声明时,无论obj.c在什么地方调用,这里的this都指向全局对象。

当obj在函数环境中声明时,这个this指向undefined,在非严格模式下,会自动转向全局对象**

"use strict";
var a = 20;
function foo () {
    var a = 1;
    var obj = {
        a: 10, 
        c: this.a + 20,
        fn: function () {
            return this.a;
        }
    }
    return obj.c;
}
console.log(foo()); // error

实际开发中,并不推荐这样使用this

再来看一些容易理解错误的例子,加深一下对调用者与是否独立运行的理解。

var a = 20;
var foo = {
    a: 10,
    getA: function () {
        return this.a;
    }
}
console.log(foo.getA()); // 10

var test = foo.getA;
console.log(test());  // 20

foo.getA()中,getA是调用者,他不是独立调用,被对象foo所拥有,因此它的this指向了foo。而test()作为调用者,尽管他与foo.getA的引用相同,但是它是独立调用的,因此this指向undefined,在非严格模式,自动转向全局window。

稍微修改一下代码,请自行理解。

var a = 20;
function getA() {
    return this.a;
}
var foo = {
    a: 10,
    getA: getA
}
console.log(foo.getA());  // 10
function foo() {
    console.log(this.a)
}
function active(fn) {
    fn(); // 真实调用者,为独立调用
}
var a = 20;
var obj = {
    a: 10,
    getA: foo
}
active(obj.getA);
三、使用call,apply显示指定this

JavaScript内部提供了一种机制,让我们可以自行手动设置this的指向,也就是call与apply。
所有的函数都具有这两个方法。它们除了参数略有不同,其功能完全一样。它们的第一个参数都为this将要指向的对象。

如下例所示。fn并非属于对象obj的方法,但是通过call,我们将fn内部的this绑定为obj,因此就可以使用this.a访问obj的a属性了。这就是call/apply的用法。

function fn() {
    console.log(this.a);
}
var obj = {
    a: 20
}
fn.call(obj);

而call与apply后面的参数,都是向将要执行的函数传递参数。其中call以一个一个的形式传递,apply以数组的形式传递。这是他们唯一的不同。

function fn(num1, num2) {
    console.log(this.a + num1 + num2);
}
var obj = {
    a: 20
}
fn.call(obj, 100, 10); // 130
fn.apply(obj, [20, 10]); // 50

因为call / apply的存在,这让JavaScript变得十分灵活。因此就让call / apply拥有了很多有用处的场景。简单总结几点,也欢迎大家补充。
1.将类数组对象转换为数组

function exam(a, b, c, d, e) {
    console.log(arguments); // { "0": 2, "1": 8, "2": 9, "3": 10, "4": 3 }
    var arg = [].slice.call(arguments);
    console.log(arg); // [ 2, 8, 9, 10, 3 ]
}
exam(2, 8, 9, 10, 3);

// 也常常使用该方法将DOM中的nodelist转换为数组
// [].slice.call( document.getElementsByTagName("li") );

2.根据自己的需要灵活修改this指向

var foo = {
    name: "joker",
    showName: function() {
      console.log(this.name);
    }
}
var bar = {
    name: "rose"
}
foo.showName.call(bar) // rose

3.实现继承

// 定义父级的构造函数
var Person = function(name, age) {
    this.name = name;
    this.age  = age;
    this.gender = ["man", "woman"];
}
// 定义子类的构造函数
var Student = function(name, age, high) {
    Person.call(this, name, age);
    this.high = high;
}
Student.prototype.message = function() {
    console.log("name:"+this.name+", age:"+this.age+", high:"+this.high+", gender:"+this.gender[0]+";");
}
new Student("xiaoming", 12, "150cm").message(); // { name:xiaom, age:12, high:150cm, gender:man }

在Student的构造函数中,借助call方法,将父级的构造函数执行了一次,相当于将Person中的代码,在Sudent中复制了一份,其中的this指向为从Student中new出来的实例对象。call方法保证了this的指向正确,因此就相当于实现了基层。Student的构造函数等同于下。

var Student = function(name, age, high) {
    this.name = name;
    this.age  = age;
    this.gender = ["man", "woman"];
    // Person.call(this, name, age); 这一句话,相当于上面三句话,因此实现了继承
    this.high = high;
}

4、在向其他执行上下文的传递中,确保this的指向保持不变

如下面的例子中,我们期待的是getA被obj调用时,this指向obj,但是由于匿名函数的存在导致了this指向的丢失,在这个匿名函数中this指向了全局,因此我们需要想一些办法找回正确的this指向。

var obj = {
    a: 20,
    getA: function() {
        setTimeout(function() {
            console.log(this.a)
        }, 1000)
    }
}
obj.getA(); // undefined

常规的解决办法很简单,就是使用一个变量,将this的引用保存起来。我们常常会用到这方法,但是我们也要借助上面讲到过的知识,来判断this是否在传递中被修改了,如果没有被修改,就没有必要这样使用了。

var obj = {
    a: 20,
    getA: function() {
        var self = this;
        setTimeout(function() {
            console.log(self.a)
        }, 1000)
    }
}

另外就是借助闭包与apply方法,封装一个bind方法。

function bind(fn, obj) {
    return function() {
        return fn.apply(obj, arguments);
    }
}
var obj = {
    a: 20,
    getA: function() {
        setTimeout(bind(function() {
            console.log(this.a)
        }, this), 1000)
    }
}
obj.getA(); // 20

当然,也可以使用ES5中已经自带的bind方法。它与上面封装的bind方法是一样的效果。

var obj = {
    a: 20,
    getA: function() {
        setTimeout(function() {
            console.log(this.a)
        }.bind(this), 1000)
    }
}
四、构造函数与原型方法上的this

在封装对象的时候,我们几乎都会用到this,但是,只有少数人搞明白了在这个过程中的this指向,就算我们理解了原型,也不一定理解了this。所以这一部分,将会为这篇文章最重要最核心的部分。理解了这里,将会对你学习JS面向对象产生巨大的帮助。

结合下面的例子大家思考一下。

function Person(name, age) {
    // 这里的this指向了谁?
    this.name = name;
    this.age = age;
}
Person.prototype.getName = function() {
    // 这里的this又指向了谁?
    return this.name;
}
// 上面的2个this,是同一个吗,他们是否指向了原型对象?
var p1 = new Person("Nick", 20);
p1.getName();

我们已经知道,this是在函数调用过程中确定,因此,搞明白new的过程中到底发生了什么就变得十分重要。

通过new操作符调用构造函数,会经历以下4个阶段。

创建一个新的对象;

将构造函数的this指向这个新对象;

指向构造函数的代码,为这个对象添加属性,方法等;

返回新对象。

因此,当调用new操作符构造函数时,this其实指向的是这个新创建的对象,最后又将新的对象返回出来,被实例对象p1接收。因此,我们可以说,这个时候,构造函数的this,指向了新的实例对象,p1。

而原型方法上的this就好理解多了,根据上边对函数中this的定义,p1.getName()中的getName为调用者,他被p1所拥有,因此getName中的this,也是指向了p1。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/81733.html

相关文章

  • 前端基础进阶(五):方位解读this

    摘要:的构造函数等同于下。通过操作符调用构造函数,会经历以下个阶段。创建一个新的对象将构造函数的指向这个新对象指向构造函数的代码,为这个对象添加属性,方法等返回新对象。前端基础进阶系列目录 showImg(https://segmentfault.com/img/remote/1460000008353088); 我们在学习JavaScript的过程中,由于对一些概念理解得不是很清楚,但是又...

    superPershing 评论0 收藏0
  • 分布式系统关注点——360°方位解读「缓存」

    摘要:在一个成熟的系统中,能够运用到缓存的地方其实并不是一处。那么在以终端用户为起点,系统所用的数据库为终点的这条道路上可以作为缓存设立点的位置大致有以下这些。缓存也有一系列的副作用需要考虑。 如果这是第二次看到我的文章,欢迎文末扫码订阅我个人的公众号(跨界架构师)哟~ 本文长度为3578字,建议阅读10分钟。坚持原创,每一篇都是用心之作~ 此前的「伸缩性」章节结束了,此文是「高性能」章...

    alanoddsoff 评论0 收藏0
  • 文森特系统用深度学习将涂鸦变成艺术创作

    摘要:研究者创建了一个名叫文森特的系统是的,就是梵高那个文森特使用深度学习,将简笔画转变为艺术品。研究人员认为,除了在艺术绘画方面大放异彩,类似文森特的技术还有一系列潜在的应用。 如果你喜欢艺术但下笔皆为灵魂画作,那么今天要介绍的这个项目肯定合你心意。AI研究者创建了一个名叫文森特(Vincet)的系统——是的,就是梵高那个文森特——使用深度学习,将简笔画转变为艺术品。用户在平板上的涂鸦经过文森特...

    PumpkinDylan 评论0 收藏0
  • 深入解读:获Forrester大数据能力高评价的阿里云DataWorks思路与能力

    摘要:阿里云成为唯一入选的中国产品。在阿里云的众多产品中,和共同构成了服务能力的核心。作为大数据能力赋能的重要手段,出现在了等阿里云专有云解决方案中。利用云计算技术,互联网公司得以快速的将自身的大数据处理能力对外赋能。 1.前言 本文基于Now Tech: Cloud Data Warehouse, Q1 2018 (Published: by Noel Yuhanna, March 13,...

    ashe 评论0 收藏0
  • 深入解读:获Forrester大数据能力高评价的阿里云DataWorks思路与能力

    摘要:阿里云成为唯一入选的中国产品。在阿里云的众多产品中,和共同构成了服务能力的核心。作为大数据能力赋能的重要手段,出现在了等阿里云专有云解决方案中。利用云计算技术,互联网公司得以快速的将自身的大数据处理能力对外赋能。 1.前言 本文基于Now Tech: Cloud Data Warehouse, Q1 2018 (Published: by Noel Yuhanna, March 13,...

    caoym 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<