谈一谈我如何使用call、apply、bind

其实我接触前端时间并不长,很多都处于学习阶段。但是随着自己掌握知识越发复杂起来,我发现学习一门新技术并没有想象中的容易,甚至要比想象中难很多。虽然很多技术方面、思想都是触类旁通的。但是很多技术深处的差异性和基础让我学的东西越来越多。

这三个名词,如果你看源码看的比较多的话。你会看到很多地方用这个东西。但是并不太了解他们的关系。我们今天就是为了彻底搞懂这三个的区别。

bind和apply

首先来谈谈call()apply(),首先他们俩都是为了改变某个函数运行时的上下文而存在的。通俗点说就是改变this的指向的。

首先我们得明白两点:

  • call()apply()是Function对象的方法,每个函数都能调用。
  • call()和apply()的第一个参数都是你要指定的执行上下文,剩下的部分就是你想要传递的参数,也就是传给调用call()和apply()方法的函数的参数。

来看下面一段代码:

function show(...arg){
    console.log(arg);
    console.log(this.name)
}
var person={
    name:"klivitam",
    age:24  
};
show.call(person,"男","爱好唱歌","宅男");

show.apply(person,["男","爱好唱歌","宅男"]);

效果图如下:


由上面的例子我们可以清楚发现,call()和apply()函数的第一个参数都是改变this的指向,而call()和apply的区别仅仅只有call()需要一项一项的传,但是apply()直接传递一个数组就行了。

apply的用法

关于call()和apply()的用法呢?我也简单介绍几个(百度上面看的,自己也实现了一下,挺好用的,但是看zepto源码里面用apply()挺多的):

  • 数组之间追加
var array1=[1,3,5];
var array2=[54,"join"];
[].push.apply(array1,array2);
console.log(array1); // [1,3,5,54,join]

这里可以理解成push方法没有提供push一个数组,但是它提供了push(param1,param,…paramN) 所以我们也可以通过apply来装拯救这个数组,因为apply会打散一个数组,一个个传进来。
也可以写成array1.push.apply(array1,array2);

  • 获取数组中的最大值和最小值
var  numbers = [1,13,-3];
var maxInNumbers = Math.max.apply(Math, numbers),   //13
maxInNumbers = Math.max.call(Math,1,13,-3); //13

number 本身没有 max 方法,但是 Math 有,我们就可以借助 call 或者 apply 使用其方法。但是用apply记得是传入一个数组,apply()里面第一个参数可以是Math,也可以是null,这个是因为没有对象去调用这个方法,我只需要用这个方法帮我运算,得到返回的结果就行。

但是它支持Math.max(param1,param2,param3…),所以可以根据刚才apply的那个特点来解决 var maxInNumbers=Math.max.apply(null,numbers),这样轻易的可以得到一个数组中最大的一项(apply会将一个数组装换为一个参数接一个参数的传递给方法,记住)

  • 验证是否是数组(前提是toString()方法没有被重写过)
// isArray的源码
function isArray(obj){ 
    return Object.prototype.toString.call(obj) === '[object Array]' ;
}

bind

bind() 方法与 apply 和 call 很相似,也是可以改变函数体内 this 的指向。来看下面一段代码

function show(){
    console.log(this.name)
}
var person={
    name:"klivitam",
    age:24  
};


show(); // undefined

var a = show.bind(person);
a(); // klivitam

由上面的代码我们可以看出bind前的show()函数打印的是undefined,因为this指向的是window,而window里面并没有name的变量。所以返回的undefined。而a()经过bind函数之后,this指向编程person,所以输出是klivitam。

bind,apply,call的共同和不同点:

  • 三者都是用来改变函数的this对象的指向的;
  • 三者第一个参数都是this要指向的对象,也就是想指定的上下文,上下文就是指调用函数的那个对象。(点前的那个对象,没有就是全局window)
  • 三者都可以传参,但是apply是数组,而call是有顺序的传入。
  • bind 是返回对应函数,便于稍后调用;apply 、call 则是立即执行

说在最后

刚刚看完了一场比赛,临时赶的一篇文章。写的不怎么好,但其实描述清楚了三者之间的区别。这周又过去了,希望下周是重整旗鼓的一周.

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

推荐阅读更多精彩内容