JavaScript中函数中的this到底是什么

目录

  1. 函数中this是什么
  2. 如何改变this的指向(call, apply, bind)
  3. 上述方法的不同点
  4. bind方法的实现原理

正文

1. 函数中this是什么

this,是指向当前函数的运行环境,也就是执行上下文。

  1. 当直接在浏览器全局环境中调用函数,那么,this指向的就是Window对象;
function simple() {
    console.log(this);
}
// 直接在全局找那个调用
simple();
全局中调用函数中的this
  1. 直接调用特定对象中的方法时,此时这个方法就运行在这个特定对象的中;
var runEnv = {
    name: 'yyp',
    age: 18,

    printThis: function() {
        console.log(this);
    }
}

// 运行runEnv对象中printThis方法
runEnv.printThis();
特定对象中方法的this
  1. 特殊情况:如果我们在全局中保留了特定对象中方法的引用后, 直接在全局中执行,这是,由于此时的方法,是在全局环境中执行的,因此this指向全局;
var runEnv = {
    name: 'yyp',
    age: 18,

    printThis: function() {
        console.log(this);
    }
}

// 在全局中保留runEnv对象中printThis方法的引用
var cloneFun = runEnv.printThis;

// 在全局中运行这个函数
cloneFun();
在全局中运行保存的新引用
  1. 使用new进行调用
    使用new调用情况则有所不同,首先会先创建一个新对象,然后将函数的this指向指向这个新对象,然后函数中所有语句都是在这个运行作用域上执行,最后返回这个新对象。
function Person(name, age) {
    this.name = name;
    this.age = age;
}
// 调用Person构造函数
var person = new Person('yyp', 12);
刚进如函数时this和执行完两条语句后的this

从上面四种情况可以得出,this在一个动态的概念,相对于运行过程的。

2. 如何改变this的指向(call, apply, bind)

以上,是this的一些常规指向情况,在实际开发中,可以根据实际的需求,改变函数中的this指向。call, apply, bind都是函数原型上的方法,因此每个函数都可以调用这些方法。

2 使用方法简介
  1. call方法
    语法: fun.call(thisArg[, arg1[, arg2[, ...]]])
    thisArg:运行时this的指向
    arg1, arg2, ...: 可选传递给函数的参数
var person1 = {
    name: 'yyp'
}

var person2 = {
    name: 'wg'
}

function sayHi() {
    console.log('Hi, ' + this.name);
}

// 绑定this指向person1
sayHi.call(person1);

// 绑定this指向person2
sayHi.call(person2);
最终运行结果
  1. apply方法
    语法: fun.apply(thisArg, [argsArray])
    thisArg:运行时this的指向
    argsArray: 传递给函数的参数列表数组
var person1 = {
    name: 'yyp'
}

var person2 = {
    name: 'wg'
}

function sayHi() {
    console.log('Hi, ' + this.name);
}

// 绑定this指向person1
sayHi.apply(person1);

// 绑定this指向person2
sayHi.apply(person2);

上面列出的代码,和上一次的代码只是将call方法变为apply,具体apply和call用户不同,下文会单独进行解释。

  1. bind方法
    语法: fun.bind(thisArg[, arg1[, arg2[, ...]]]);
    thisArg:运行时this的指向
    arg1, arg2, ...: 可选传递给函数的参数
var person1 = {
    name: 'yyp'
}

var person2 = {
    name: 'wg'
}

function sayHi() {
    console.log('Hi, ' + this.name);
}

// 绑定this指向person1,返回一个函数
var hello_person1 = sayHi.bind(person1);
// 执行函数
hello_person1();

// 绑定this指向person2,返回一个函数
var hello_person2 = sayHi.bind(person2);
// 执行函数
hello_person2();

最终的运行结果和上面两种情况中一样。

3. 上述方法的不同点
  1. call和apply方法的不同之处
    call,apply方法主要是在传递参数的方式上不同,上面举例的函数不需要提供参数,因此,两个可以交替使用。
    但是当函数调用需要传递参数时,两者的使用方法就不一样了,有上面提供的语法格式可以知道,call方法将需要传递的参数平铺,一个一个的传递,而apply方法,则需要将所有需要传递的参数放到一个数组中,然后传递给函数
var runEnv = {
    name: 'yyp'
}

function sayHi(str) {
    console.log(str + ' ' + this.name);
}

// 使用call方法绑定this,并传递参数,将参数依次传递
sayHi.call(runEnv, 'Hi!');

// 使用bind方法绑定this,并传递参数,所有传入的参数,需要先组成数组,然后作为第二个参数传入
sayHi.apply(runEnv, ['Hello!']);

在使用过程中如何选择
1.1 如果给定参数列表是数组形式,选用apply

var items = [1, 2, 3, 4];

// Math对象中的max方法,可以接受多个参数
// 如果需要梳理的元素是一个数组,那么我们可以使用apply使用方法
// 不用将数组中的元素一个个提取出来,在进行处理
var max = Math.max.apply(null, items);

console.log(max);

1.2 从性能方面考虑

ECMA-262文档中call方法定义
ECMA-262文档中apply方法定义

从这两个文档中我们可以发现,调用call方法,如果传入参数,直接从第二个参数从左向右将参数添加到argList中即可,而apply方法要调用一个CreateListFromArrayLike方法,将传入的数组元素处理为合法的argList,因此在可能存在性能上的差异。

jsPerf中对其运行性能进行对比发现对比结果。在参数较少(1-3)个时采用call的方式调用(lodash就是采用这种方式重写的)。

  1. bind方法和其他两种方法的不同之处
    bind方法,是ES5才提出的,其主要的不同就在于,call和apply方法,是在指定的作用域上直接运行函数,而bind方法是创建一个新的函数供后期直接使用,其传入的参数,也会直接绑定到这个新函数上。

2.1 bind函数的运行作用域
即使在全局作用域中运行,或者用call或者apply绑定其他作用域,都不会改变其运行作用域。

function Person(name, age) {
    this.name = name;
    this.age = age;
}

// 创建一个空对象
var emptyObj = {};
// 将函数的this指向emptyObj,获取一个新的函数
var bindPerson = Person.bind(emptyObj);
// 在全局中运行该函数
bindPerson('yyp', 18);
运行后emptyObj对象内容

运行绑定后获取的新函数时,此时我们已经将this绑定到制定的空对象上,运行bindPerson函数,相当于运行Person.call(emptyObj, 'yyp', 18);因此会在emptyObj上添加两个属性name和age。

2.2 bind时绑定提前绑定参数
如果在调用bind函数时,提供传入参数时,此时这些参数,将会直接绑定到新函数上,后续执行新函数传入的参数将会追加到这些参数后面。

function Person(name, age) {
    this.name = name;
    this.age = age;
}

var emptyObj = {};
// 此时提供一个参数
var bindPerson = Person.bind(emptyObj, 'yyp');
// 调用时提供一个参数,此时效果和前一个保持一致
bindPerson(18);

// 提供两个参数时,根据实际参数情况,会丢掉后面的参数
// bindPerson('yyp', 18);

2.3 使用new调用函数时,出现的问题
使用new函数进行调用时,this不会指向bind时设定的对象,而是和直接使用new调用原始函数行为保持一致,但是之前提前绑定的参数,还是会生效。

function Person(name, age) {
    this.name = name;
    this.age = age;
}

var emptyObj = {};
var bindPerson = Person.bind(emptyObj);

var person = new bindPerson('yyp', 18);
使用new调用函数时运行结果

从上面运行结果可以发现,使用new调用时,绑定的作用于失效,this原先的this,因此emptyObj还是为空。

4. bind方法的实现原理
Function.prototype.bind = function(oThis) {
    // 判断调用这个方法的对象是不是一个函数
    if (typeof this !== 'function') {
        // closest thing possible to the ECMAScript 5
        // internal IsCallable function
        throw new TypeError('Function.prototype.bind - what is trying to be bound is not callable');
    }
    // 处理传入的参数部分
    // aArgs:保存绑定时传入的参数
    // fToBind:指向需要绑定的函数
    // fNOP: 空函数
    // fBound:将要返回的函数引用
    var aArgs = Array.prototype.slice.call(arguments, 1),
        fToBind = this,
        fNOP = function() {},
        fBound = function() {
            // 绑定函数执行时运行的处理逻辑
            // 如果当前任何环境中运行,执行函数中的this为之前制定的作用域,即作用域不会做二次绑定
            // 如果使用new进行调用时,执行函数中的this不改变
            return fToBind.apply(this instanceof fNOP ?
                this :
                oThis,
                // 获取调用时(fBound)的传参.bind 返回的函数入参往往是这么传递的
                aArgs.concat(Array.prototype.slice.call(arguments)));
        };

    // 维护原型关系
    if (this.prototype) {
        // Function.prototype doesn't have a prototype property
        fNOP.prototype = this.prototype;
    }

    // 返回的函数fBound继承fNOP,是fNOP的一个实例,
    // 作用:1. 维持原型链
    // 2. 用new进行该函数调用时,此时的this是fNOP的一个实例
    fBound.prototype = new fNOP();

    return fBound;
};

由上面代码可以了解到,调用bind函数进行绑定后,会返回一个新的函数,并且将调用时传入的参数保存起来。当调用这个绑定函数时,先判断this的指向,如果是使用new调用,this将会是bind函数的实例(绑定函数又是内部fNOP的实例),直接使用当前this;如果是其他情况,则直接在之前绑定的运行作用域上执行。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容