JavaScript深入之bind的模擬實(shí)現(xiàn)

JavaScript深入之bind的模擬實(shí)現(xiàn)

bind() 方法會(huì)創(chuàng)建一個(gè)函數(shù),當(dāng)這個(gè)新函數(shù)被調(diào)用的時(shí)候,bind() 的第一個(gè)參數(shù)將作為他運(yùn)行時(shí)的this,之后的一序列參數(shù)將會(huì)在傳遞的參數(shù)前作為他的參數(shù)

其實(shí)這也可以看出bind函數(shù)的兩個(gè)特點(diǎn)

  • 返回一個(gè)新函數(shù)
  • 可以傳入?yún)?shù)

先看一個(gè)例子----返回函數(shù)的實(shí)現(xiàn)

var foo = {
    value : 2
}
function bar(){
    console.log(this.value)
}
var bindBar = bar.bind(foo)
bindBar();
// 2

我們定義的bind2 的函數(shù)要定義在Function.prototype,這是因?yàn)槲覀兒瘮?shù)都是由Function創(chuàng)建的,window也是Function.prototype的實(shí)例

window.__proto__.__proto__.__proto__.__proto__.__proto__ == Object.prototype.__proto__
//true
Object.__proto__  == Function.prototype
//true

window.__proto__.__proto__.__proto__.__proto__ == Function.prototype.__proto__
//true
Function.prototype.bind2 = function(context){
    console.log(111, this)
    //這個(gè)this就指向調(diào)用者。如果是bar函數(shù),就指向bar函數(shù)
    //context表示的是傳入的對(duì)象 也就是說
    //bar函數(shù)中的this指向context中的this
    
    var self = this
    return function(){
    self.apply(context)
    }
}

var foo = {
    value:3
}
function bar(){
    console.log(this.value)
}
var bindFoo = bar.bind2(foo);
//bind2 返回的只是bind2中的新函數(shù),所以需要再調(diào)用一次
console.log(bindFoo()) //3

傳參的模擬實(shí)現(xiàn)

先看騷騷的bind傳參,繼續(xù)剛才的騷例子

var foos = {
    value:4
}
function foo(name,age){
    console.log(name)
    console.log(age)
    console.log(this.value)
}

var bindFoo = foo.bind(foos, 'xiaolizi')
bindFoo(18)
// xiaolizi
//18
// 4

foo需要傳入name和age兩個(gè)參數(shù),而且可以在bind的時(shí)候,只傳一個(gè)name,然后再執(zhí)行返回的函數(shù)的時(shí)候再傳入另一個(gè)參數(shù)age
所以我們?cè)俚诙嫘枰獙?duì)arguments進(jìn)行處理

第二版
Function.prototype.bind2 = function(context){
    var self = this;
//因?yàn)閍rguments是類數(shù)組對(duì)象,不能使用slice,所以使用call改變this的指向
//從slice(1) 是截取第一個(gè)參數(shù)之后的參數(shù),因?yàn)榈谝粋€(gè)參數(shù)是傳入的對(duì)象context。
    var args = Array.prototype.slice.call(arguments,1)
    return function(){
    //這個(gè)是截取第二個(gè)調(diào)用的函數(shù),所以截取全部的參數(shù)
        var bindArgs = Array.prototype.slice.call(arguments)
        return self.apply(context,args.concat(bindArgs))
        }
}

var foos = {
    value:4
}
function foo(name,age){
    console.log(name)
    console.log(age)
    console.log(this.value)
}

var bindFoo = foo.bind2(foos, 'xiaolizi')
bindFoo(18)

構(gòu)造函數(shù)效果的模擬實(shí)現(xiàn)

一個(gè)綁定函數(shù)也能使用new操作符創(chuàng)建對(duì)象,這種行為就像把函數(shù)當(dāng)做構(gòu)造器,提供的this值被忽略,同時(shí)調(diào)用時(shí)的參數(shù)被提供給模擬函數(shù)

其實(shí)也就是說bind返回的函數(shù)作為構(gòu)造函數(shù)的時(shí)候,bind時(shí)指定的this值會(huì)失效,但傳入的參數(shù)依然有效。

var value = 2;

var foo = {
    value: 1
};

function bar(name, age) {
    this.habit = 'shopping';
    console.log(this.value);
    console.log(name);
    console.log(age);
}

bar.prototype.friend = 'kevin';

var bindFoo = bar.bind(foo, 'daisy');

var obj = new bindFoo('18');
// undefined
// daisy
// 18
console.log(obj.habit);
console.log(obj.friend);
// shopping
// kevin

注意:盡管在全局和 foo 中都聲明了 value 值,最后依然返回了 undefind,說明綁定的 this 失效了,如果大家了解 new 的模擬實(shí)現(xiàn),就會(huì)知道這個(gè)時(shí)候的 this 已經(jīng)指向了 obj。

// 第三版
Function.prototype.bind2 = function (context) {
    var self = this;
    var args = Array.prototype.slice.call(arguments, 1);

    var fBound = function () {
        var bindArgs = Array.prototype.slice.call(arguments);
        // 當(dāng)作為構(gòu)造函數(shù)時(shí),this 指向?qū)嵗藭r(shí)結(jié)果為 true,將綁定函數(shù)的 this 指向該實(shí)例,可以讓實(shí)例獲得來自綁定函數(shù)的值
        // 以上面的是 demo 為例,如果改成 `this instanceof fBound ? null : context`,實(shí)例只是一個(gè)空對(duì)象,將 null 改成 this ,實(shí)例會(huì)具有 habit 屬性
        // 當(dāng)作為普通函數(shù)時(shí),this 指向 window,此時(shí)結(jié)果為 false,將綁定函數(shù)的 this 指向 context
        return self.apply(this instanceof fBound ? this : context, args.concat(bindArgs));
    }
    // 修改返回函數(shù)的 prototype 為綁定函數(shù)的 prototype,實(shí)例就可以繼承綁定函數(shù)的原型中的值
//這里實(shí)際吧fBound 作為function,也就是第一調(diào)用的時(shí)候就返回一個(gè)函數(shù),函數(shù)里面再根據(jù)調(diào)用的方式,確定this的指向,當(dāng)?shù)诙沃赶虻臅r(shí)候,this指向的開始改變并賦值給fBound.prototype 就可以改變this的指向
    fBound.prototype = this.prototype;
    return fBound;
}

構(gòu)造函數(shù)效果的優(yōu)化實(shí)現(xiàn)

但是在這個(gè)寫法中,我們直接將 fBound.prototype = this.prototype,我們直接修改 fBound.prototype 的時(shí)候,也會(huì)直接修改綁定函數(shù)的 prototype。這個(gè)時(shí)候,我們可以通過一個(gè)空函數(shù)來進(jìn)行中轉(zhuǎn):

// 第四版
Function.prototype.bind2 = function (context) {

    var self = this;
    var args = Array.prototype.slice.call(arguments, 1);

    var fNOP = function () {};

    var fBound = function () {
        var bindArgs = Array.prototype.slice.call(arguments);
        return self.apply(this instanceof fNOP ? this : context, args.concat(bindArgs));
    }

    fNOP.prototype = this.prototype;
    fBound.prototype = new fNOP();
    return fBound;
}

最終版

Function.prototype.bind2 = function (context) {

    if (typeof this !== "function") {
      throw new Error("Function.prototype.bind - what is trying to be bound is not callable");
    }

    var self = this;
    var args = Array.prototype.slice.call(arguments, 1);

    var fNOP = function () {};

    var fBound = function () {
        var bindArgs = Array.prototype.slice.call(arguments);
        return self.apply(this instanceof fNOP ? this : context, args.concat(bindArgs));
    }

    fNOP.prototype = this.prototype;
    fBound.prototype = new fNOP();
    return fBound;
}

以上參考一下文章加了一些自己的理解

JavaScript深入之bind的模擬實(shí)現(xiàn)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,363評(píng)論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,497評(píng)論 3 416
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,305評(píng)論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,962評(píng)論 1 311
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,727評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,193評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,257評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,411評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,945評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,777評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,978評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,519評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,216評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評(píng)論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,657評(píng)論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,960評(píng)論 2 373

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