2. this、call和apply

本文源于本人关于《JavaScript设计模式与开发实践》(曾探著)的阅读总结。想详细了解具体内容建议阅读该书。

1. this

this总是指向一个对象,而具体指向哪个对象是在运行时基于函数的执行环境动态绑定的。

this的指向

除去with和eval,实际应用中,this指向分为以下四类:

  • 作为对象的方法调用
  • 作为普通函数调用
  • 构造器调用
  • Function.prototype.call & Function.prototype.apply
作为对象的方法调用
var obj = {
    a: 1,
    getA: function(){
        console.log(this === obj); // true
        console.log(this.a); // 1
    }
}

obj.getA();

这个时候,由于是obj调用的getA()函数,故this指向obj。

作为普通函数调用
var obj = {
    name: aaaa,
    getName: function(){
        console.log(this === obj); // false
        console.log(this.name); // yozo
    }
}

window.name = yozo;
var getGlobalName = function(){
    console.log(this.name);
}

var getName = obj.getName;

getGlobalName(); // yozo;
getName();

只要函数没有调用者,则this就指向全局对象,在浏览器中的Js里,全局对象是window。严格模式中,this不指向全局对象。
这里主要看getName,它本身只是一个函数,由于被抽离出来了,故执行getName()时,它是没有调用者的,和getGlobalName一样,作为普通函数调用。

构造器调用
var Myclass = function(){
    this.name = 'yozo';
};

var obj = new Myclass();
console.log(obj.name); // yozo

这个时候的this代表被新创建出来的那个对象,这里指obj

但是:

var Myclass = function(){
    this.name = 'yozo';
    return {
        name: 'ann'
    }
};

var obj = new Myclass();
console.log(obj.name); // ann

当显示返回一个对象时,那么这个时候显示返回的这个对象会被obj引用,故这个时候 obj.name就是ann而不是yozo。

Function.prototype.call & Function.prototype.apply
var obj1 = {
    name: 'yozo',
    getName: function(){
        console.log(this.name);
    }
}

var obj2 = {
    name: 'ann'
}

console.log(obj1.getName()); // yozo
console.log(obj1.getName.call(obj2)); // ann
  • 第一个console中,getName的调用者是obj1,则this指向obj1;
  • 第二个console中,getName的调用者本是obj1,但是这个时候出现了.call(obj2)。就好像是打电话告诉obj2:“喂,obj2,你去执行obj1的getName方法吧”,故这个时候调用者就变成了obj2,我们知道,this总是指向调用者,故这个时候this就指代obj2。

2. call和apply

call和apply的区别

刚刚我们已经说了call的作用了,就是改变函数的调用者。那么apply的作用呢?也是改变函数的调用者。区别在于,call和apply都是Function.prototype上的方法,故只有函数才能调用这两个函数, 既然是函数的call和apply,那么函数在执行的时候总会有参数吧,那么改变了调用者之后,函数的参数怎么处理?

  • call: 把参数一个一个列举出来(不固定参数,第一个为新的调用者,其他的为传入该函数的其他参数)
  • apply: 把参数一锅传(只有两个参数,第一个为新的调用者,另一个参数为参数数组)
var func = function(a, b, c){
    console.log([a, b, c]);
}

func.apply(null,[1, 2, 3]);
func.call(null, 1, 2, 3);

这个函数没this,所以不需要调用者,我们只用来区别call和apply的用法:

  • call:从第二个参数开始,一一对应func的各个参数
  • apply:参数数组与func的各个参数一一对应。
call和apply用途
  • 改变this指向(改变调用者)
document.getElementById('div1').onclick = function(){
    console.log(this.id); // div1
    var func = function(){
        console.log(this.id);
    }
    func(); // undefined (无调用者)
    func.call(this); // div1 
}

onclick回调函数中this表示document.getElementById('div1'), 但是执行func()的时候没有调用者,默认调用者为window,但是我们希望它仍能保持指向document.getElementById('div1'),则把this传给了func,让他调用该函数。

  • Function.prototype.bind:将一个函数的this固定为某个对象,之后使用时就不用再使用call和apply了。
    简易版:
Function.prototype.bind = function (context) {
  var self = this;
  return function () {
    self.apply(context, arguments);
  }
}

var obj1 = {
  name: 'yozo',
  getName: function () {
    console.log(this.name);
  }
}

var obj2 = {
  name: 'ann'
}

var obj2getName = obj1.getName.bind(obj2);
obj2getName(); // ann

这时的this就已经被固定为obj2了。

这个版本在固定对象时不能保存参数,故升级版:

Function.prototype.bind = function () {
  var self = this; // 保留该函数的调用者
  var context = [].shift.call(arguments); // 获取参数数组中第一个参数,即需要被固定的那个对象
  var args = [].slice.call(arguments); // 保存剩下的参数
  
  // 返回一个新的函数
  return function () {
    // 改变调用者为需要被固定的那个对象,将原本剩下的参数,和新传入的参数组成一个参数数组并执行
    self.apply(context, [].concat.call(args, [].slice.call(arguments)));
  }
}

var obj = {
  name: 'yozo'
}

var func = function (a, b, c, d) {
  console.log(this.name);
  console.log([a, b, c, d]);
}.bind(obj, 1, 2);

func(3, 4); // 1, 2, 3, 4
  • 借用其他对象的方法:之前的obj1.getName.call(obj2)就是obj2借用obj1的方法。我们常用的是借用Object或者Array的prototype的方法:
(function(){
    Array.prototype.push.call(arguments, 3, 4);
    console.log(arguments);
})(1, 2)

// 1, 2, 3, 4

这其实是arguments参数数组对象利用了真数组对象,参数数组为伪数组(DOM节点数组也是),但是伪数组可以借用真数组的方法。

伪数组需要满足两个条件:

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

推荐阅读更多精彩内容