一次性講清楚apply/call/bind

說到apply/call/bind,必須首先提一下this竞漾,因為這三者的根本作用就是改變函數(shù)運行時的this指向眯搭,所以要想說清它們塑煎,必須先說清this

以下是函數(shù)的調(diào)用方法:

  • 方法調(diào)用模式:當(dāng)一個函數(shù)被保存為對象的一個屬性時底哗,我們稱它為對象的一個方法,如果調(diào)用表達式包含一個提取屬性的動作峭判,那么它就是被當(dāng)做一個方法來調(diào)用,此時的this被綁定到這個對象笔时。
    var a = 1
    var obj1 = {
      a:2,
      fn:function(){
        console.log(this.a)
      }
    }
    obj1.fn()//2    

此時的this是指obj1這個對象棍好,事實上誰調(diào)用這個函數(shù),this就是誰,補充一下允耿,DOM對象綁定事件也屬于方法調(diào)用模式梳玫,因此它綁定的this就是事件源DOM對象

  • 函數(shù)調(diào)用模式:就是普通函數(shù)的調(diào)用,此時的this被綁定到window
  1. 最最普通的函數(shù)調(diào)用
function fn1(){
      console.log(this)//window
    }
fn1()
  1. 函數(shù)嵌套
function fn1(){
    function fn2(){
        console.log(this)//window
    }
    fn2()
}
fn1()
  1. 把函數(shù)賦值之后再調(diào)用
var a = 1
var obj1 = {
    a:2,
    fn:function(){
        console.log(this.a)
    }
}
var fn1 = obj1.fn
fn1()//1

此時fn1就是不帶任何修飾的函數(shù)調(diào)用右犹,因此它的this綁定的就是window提澎,它也被稱為隱性綁定

  1. 回調(diào)函數(shù)
    這上面的函數(shù)是一個回調(diào)函數(shù),先簡單的拆分一下念链,
var a = 1
function f1(fn){
    fn()
    console.log(a)//1
}
f1(f2)

function f2(){
    var a = 2
}

簡單分析一下f1的運行,

//參數(shù)的隱性賦值
var fn = undefined
fn = f2
fn()

有沒有很熟悉的感覺盼忌,其實這又回到了第三種情況,因此最后的a打印出的是全局的a掂墓,因此回調(diào)函數(shù)綁定的也是全局的this谦纱,借此,我們終于可以解釋為什么setTimeout總是丟失this了君编,因為它也就是一個回調(diào)函數(shù)跨嘉,而已。

setTimeout(function() {
    console.log(this)//window
    function fn(){
        console.log(this)//window
    }
    fn()
}, 0);
  • 構(gòu)造器調(diào)用模式:如果一個函數(shù)前面帶上new來調(diào)用吃嘿,那么背地里會將創(chuàng)建一個連接到prototype成員的新對象祠乃,同時this會被綁定到那個新對象上
function Person(name,age){
// 這里的this都指向?qū)嵗?    this.name = name
    this.age = age
    this.sayAge = function(){
        console.log(this.age)
    }
}

var dcbryant = new Person('dcbryant',22)
dcbryant.sayAge()//22
  • Apply調(diào)用模式
    說了那么多,終于提到我們的主角apply了兑燥,apply方法讓我們構(gòu)建一個參數(shù)數(shù)組給調(diào)用函數(shù)亮瓷,它也允許我們選擇this的值,apply接受兩個參數(shù)降瞳,第一個是要綁定給this的值嘱支,第二個就是一個參數(shù)數(shù)組。當(dāng)?shù)谝粋€參數(shù)為null挣饥、undefined的時候除师,默認指向window
var arr = [1,2,3,89,46]
var max = Math.max.apply(null,arr)//89

前面的模式this都是被指定的,而這個方法可就厲害了扔枫,由我們自己掌控的this的指向汛聚,將this指向改為apply第一個參數(shù)即可,那我們不得不懷疑一下其他的模式是不是瀏覽器自己幫我們apply的呢茧吊?所以我們又可以這么理解:

obj1.fn() 
obj1.fn.apply(obj1);

fn1()
fn1.apply(null)

f1(f2)
f1.call(null,f2)//call和apply的作用一樣贞岭,只是第二個參數(shù)稍有不同

這樣一來this的謎題終于大白于天下了,apply函數(shù)的作用就是動態(tài)的改變this上下文搓侄,那apply瞄桨、call、bind三者有什么異同呢讶踪?

先說說apply芯侥、call,其實這兩者大同小異乳讥,他們除了第二個參數(shù)有些差異柱查,其他都是一樣的,call接受的是若干個參數(shù)的列表云石,而apply接受的一個包含多個參數(shù)的數(shù)組或者類數(shù)組唉工。

再說說apply、call和bind汹忠,call和apply改變了函數(shù)的this上下文后便執(zhí)行該函數(shù),而bind則是返回改變了上下文后的一個函數(shù),靜態(tài)綁定函數(shù)執(zhí)行上下文的this屬性淋硝,并且不隨函數(shù)的調(diào)用方式而變化,知道了bind的作用,不如我們自己來實現(xiàn)一個:

Function.prototype.bind = Function.prototype.bind ||
function(context){
    var self = this
    var args = Array.prototype.slice.call(arguments, 1)
    return function(){
        var innerArgs = Array.prototype.slice.call(arguments);
        var finalArgs = args.concat(innerArgs);
        return self.apply(context,finalArgs);
    }
}

這次的bind()方法可以綁定對象宽菜,也支持在綁定的時候傳參,但是Javascript的函數(shù)還可以作為構(gòu)造函數(shù)谣膳,那么綁定后的函數(shù)用這種方式調(diào)用時,情況就比較微妙了铅乡,需要涉及到原型鏈的傳遞

Function.prototype.bind = Function.prototype.bind ||
function(context){
    var self = this;
    var args = Array.prototype.slice.call(arguments, 1);
    var F = function(){};

    var bound = function(){
        var innerArgs = Array.prototype.slice.call(arguments);
        var finalArgs = args.concat(innerArgs);
        return self.apply((this instanceof F ? this : context), finalArgs);
    };

    F.prototype = self.prototype;
    bound.prototype = new F();
    return bound
}

設(shè)置一個中轉(zhuǎn)構(gòu)造函數(shù)F继谚,使綁定后的函數(shù)與調(diào)用bind()的函數(shù)處于同一原型鏈上,用new操作符調(diào)用綁定后的函數(shù)阵幸,返回的對象也能正常使用instanceof

最后說說主要應(yīng)用場景:
1.求數(shù)組中的最大和最小值

var arr = [1,2,3,89,46]
var max = Math.max.apply(null,arr)//89
var min = Math.min.apply(null,arr)//1

2.將類數(shù)組轉(zhuǎn)化為數(shù)組

var trueArr = Array.prototype.slice.call(arrayLike)

3.數(shù)組追加

var arr1 = [1,2,3];
var arr2 = [4,5,6];
var total = [].push.apply(arr1, arr2);//6
// arr1 [1, 2, 3, 4, 5, 6]
// arr2 [4,5,6]

4.判斷變量類型

function isArray(obj){
    return Object.prototype.toString.call(obj) == '[object Array]';
}
isArray([]) // true
isArray('dcbryant') // false

5.利用call和apply做繼承

function Person(name,age){
    // 這里的this都指向?qū)嵗?    this.name = name
    this.age = age
    this.sayAge = function(){
        console.log(this.age)
    }
}
function Boy(){
    Person.apply(this,arguments)//將父元素所有方法在這里執(zhí)行一遍就繼承了
}
var dcbryant = new Boy('dcbryant',22)

6.使用 log 代理 console.log


function log(){
  console.log.apply(console, arguments);
};

參考鏈接:
js中call花履、apply、bind那些事
Javascript中bind()方法的使用與實現(xiàn)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末挚赊,一起剝皮案震驚了整個濱河市臭挽,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌咬腕,老刑警劉巖欢峰,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異涨共,居然都是意外死亡纽帖,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進店門举反,熙熙樓的掌柜王于貴愁眉苦臉地迎上來懊直,“玉大人,你說我怎么就攤上這事火鼻∈夷遥” “怎么了雕崩?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長融撞。 經(jīng)常有香客問我盼铁,道長,這世上最難降的妖魔是什么尝偎? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任饶火,我火速辦了婚禮,結(jié)果婚禮上致扯,老公的妹妹穿的比我還像新娘肤寝。我一直安慰自己,他們只是感情好抖僵,可當(dāng)我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布鲤看。 她就那樣靜靜地躺著,像睡著了一般耍群。 火紅的嫁衣襯著肌膚如雪刨摩。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天世吨,我揣著相機與錄音澡刹,去河邊找鬼。 笑死耘婚,一個胖子當(dāng)著我的面吹牛罢浇,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播沐祷,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼嚷闭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了赖临?” 一聲冷哼從身側(cè)響起胞锰,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎兢榨,沒想到半個月后嗅榕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡吵聪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年凌那,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片吟逝。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡帽蝶,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出块攒,到底是詐尸還是另有隱情励稳,我是刑警寧澤佃乘,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站驹尼,受9級特大地震影響趣避,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜扶欣,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望千扶。 院中可真熱鬧料祠,春花似錦、人聲如沸澎羞。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽妆绞。三九已至顺呕,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間括饶,已是汗流浹背株茶。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留图焰,地道東北人启盛。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓,卻偏偏與公主長得像技羔,于是被迫代替她去往敵國和親僵闯。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容