全方位解读this

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

起首,了解下实行高低文的生命周期。

《全方位解读this》

在实行高低文的竖立阶段,会分别天生变量对象,竖立作用域链,以及肯定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://segmentfault.com/a/1190000008476233
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞