成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

call,apply and bind in JavaScript

JohnLui / 2514人閱讀

摘要:文章盡量使用大量實(shí)例進(jìn)行講解,它們的使用場(chǎng)景。在嚴(yán)格模式下,函數(shù)被調(diào)用后,里面的默認(rèn)是后面通過(guò)調(diào)用函數(shù)上的和方法,該變指向,函數(shù)里面的指向。利用,可以傳入外層的上下文。同樣適用的還有,里面的對(duì)象,它也是一種類(lèi)數(shù)組對(duì)象。

call,apply and bind in JavaScript
在ECMAScript中,每個(gè)函數(shù)都包含兩個(gè)繼承而來(lái)的方法:apply() 和 call(),這兩個(gè)方法的用途都是在特定的作用域中調(diào)用函數(shù),主要作用跟bind一樣,用來(lái)改變函數(shù)體內(nèi)this的指向,或者說(shuō)是在函數(shù)調(diào)用時(shí)改變上下文。

文章盡量使用大量實(shí)例進(jìn)行講解,它們的使用場(chǎng)景。同時(shí),也會(huì)由淺入深的引導(dǎo)出一些理論,畢竟這幾個(gè)常用方法,在MDN上都能找到合理的解釋

基本功能 改變this的指向
  var fruit = {
    fruitName:"apple"
  }
  function getFruit() {
    console.log("I like "+this.fruitName)
  }

  getFruit();    // log   I like undefined
  getFruit.call(fruit)    // log   I like apple
  getFruit.apply(fruit)   // log   I like apple
  var newBind = getFruit.bind(fruit)
  newBind();              // log   I like apple

當(dāng) getFruit 并非作為一個(gè)對(duì)象的屬性,而是直接當(dāng)做一個(gè)函數(shù)來(lái)調(diào)用,里面的this就會(huì)被綁定到全局對(duì)象上,即window上, 所以直接調(diào)用 getFruit,里面的this指向了全局對(duì)象上,返回 undefined。

在嚴(yán)格模式下,函數(shù)被調(diào)用后,里面的this默認(rèn)是 undefined

后面,通過(guò)調(diào)用函數(shù)上的callapply方法,該變this指向,函數(shù)里面的this指向fruit。

區(qū)別:
bind同樣實(shí)現(xiàn)了改變this指向的功能,但是它不會(huì)立即執(zhí)行,而是會(huì)重新創(chuàng)建一個(gè)綁定函數(shù),新函數(shù)被調(diào)用時(shí),使用bind()方法里面的第一個(gè)參數(shù)作為this

接受參數(shù)

這三個(gè)方法,從接受的第二參數(shù)開(kāi)始,都直接傳遞給函數(shù),但是接受參數(shù)的方法卻很大的不同。

call,從第二個(gè)參數(shù)開(kāi)始,以參數(shù)列表的形式展示,

apply,則把傳遞的函數(shù)參數(shù),放在一個(gè)數(shù)組里面作為第二個(gè)參數(shù)。

fn.call(obj,arg1,arg2);
fn.apply(obj,[arg1,arg2])

bind,從第二個(gè)參數(shù)開(kāi)始,同樣以參數(shù)列表的形式,但是會(huì)提前放在新綁定函數(shù)的參數(shù)之前

 var foo = function(name,age){
   console.log("name: "+name+"- age: "+age)
 }

 var p1 = foo.bind(this,"popo");   // "popo" 作為新函數(shù)的第一個(gè)參數(shù)。
 p1(13);                       // logs    name: popo- age: 13
 p1("bobo",14)                 // logs    name: popo- age: bobo
應(yīng)用場(chǎng)景

綁定事件回調(diào)中

  $(".div-class").on("click",function(event) {
        /*TODO*/
        }.bind(this));
      }
  }

通常,我們?cè)诟淖兒瘮?shù)上下文之前,都會(huì)使用類(lèi)似that = this,或者self,_this,來(lái)把this賦值給一個(gè)變量。利用.bind(),可以傳入外層的上下文。

循環(huán)回調(diào)

循環(huán)中利用閉包來(lái)處理回調(diào)

for(var i = 0;i < 10;i++){
 (function(j){
     setTimeout(function(){            
         console.log(j);
     },600);
 })(i)
}

每次循環(huán),都會(huì)產(chǎn)生一個(gè)立即執(zhí)行的函數(shù),函數(shù)內(nèi)部的局部變量j保存不同時(shí)期i的值,循環(huán)過(guò)程中,setTimeout回調(diào)按順序放入消息隊(duì)列中,等f(wàn)or循環(huán)結(jié)束后,堆棧中沒(méi)有同步的代碼,就去消息隊(duì)列中,執(zhí)行對(duì)應(yīng)的回調(diào),打印出j的值。

同理,可以利用bind,每次都創(chuàng)建新的函數(shù),并且已經(jīng)預(yù)先設(shè)置了參數(shù),傳入不同的指針

  function func(i) {
    console.log(i)
  }
  for(var i =0 ;i< 10;i++) {
    setTimeout(func.bind(null,i),600)
  }

實(shí)現(xiàn)繼承

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

 var P1 = function(name,age) {
   // 借用構(gòu)造函數(shù)的方式實(shí)現(xiàn)繼承
   // 利用call 繼承了Person
   Person.call(this,name,age)
 }
 P1.prototype.getName = function() {
   console.log("name: "+this.name+", age: "+this.age);
 }

 var newPerson = new P1("popo",20);   // logs name: popo, age: 20
 newPerson.getName();

實(shí)質(zhì)上,可以看成通過(guò)call()或者apply()方法,在即將新建的對(duì)象,即這里的newPerson上,執(zhí)行超類(lèi)型的構(gòu)造函數(shù),分別在當(dāng)前上下文this上添加nameage屬性。

數(shù)組驗(yàn)證的終極方法

  function isArray(value) {
    return Object.prototype.toString.call(value) == "[object Array]"
  }

借用了Object原生的toString()方法,打印出對(duì)應(yīng)變量的構(gòu)造函數(shù)名,

類(lèi)數(shù)組轉(zhuǎn)換為數(shù)組

  // 實(shí)現(xiàn)一個(gè)簡(jiǎn)單的數(shù)組 "unshift"方法
  Array.prototype.unshift = function(){
    this.splice.apply(this,
      [0,0].concat(Array.prototype.slice.apply(arguments)));
      return this.length;
  }

首先,利用this.splice.apply(),其中splice,可以直接從數(shù)組中移除或者插入變量。apply()則以數(shù)組的形式傳遞參數(shù),需要利用concat拼接數(shù)組。

當(dāng)函數(shù)被調(diào)用時(shí),在函數(shù)內(nèi)部會(huì)得到類(lèi)數(shù)組arguments,它擁有一個(gè)length屬性,但是沒(méi)有任何數(shù)組的方法。所以,將slice方法中的this指向arguments,獲取到arguments的長(zhǎng)度,從而確定方法的startend下標(biāo),得到一個(gè)數(shù)組變量。

同樣適用的還有,DOM里面的NodeList對(duì)象,它也是一種類(lèi)數(shù)組對(duì)象。

深入理解 實(shí)現(xiàn)bind 方法

bind方法在ECMAScript5里面被引入,前面提到過(guò),調(diào)用該方法時(shí),返回一個(gè)新的函數(shù),可以簡(jiǎn)單使用下面方法實(shí)現(xiàn)其改變this指向的功能。

  Function.prototype.bind = function(scope) {
    var fn = this;
    return function() {
      return fn.apply(scope)
    }
  }

接著,就可以利用concat把bind傳遞的預(yù)置參數(shù)拼接到新函數(shù)的參數(shù)列表中。

   Function.prototype.bind = function(scope) {
      var args = Array.prototype.slice.call(arguments,1)
      var fn = this
      return function() {
        return fn.apply(scope,args.concat(Array.prototype.slice.call(arguments)))
      }
   }

參考鏈接

Javascript之bind

Understanding JavaScript Bind ()

深入淺出妙用 Javascript 中 apply、call、bind

前端基礎(chǔ)進(jìn)階:全方位解讀 this

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/93371.html

相關(guān)文章

  • 理解 JavaScript call()/apply()/bind()

    摘要:理解文章中已經(jīng)比較全面的分析了在中的指向問(wèn)題,用一句話來(lái)總結(jié)就是的指向一定是在執(zhí)行時(shí)決定的,指向被調(diào)用函數(shù)的對(duì)象。與和直接執(zhí)行原函數(shù)不同的是,返回的是一個(gè)新函數(shù)。這個(gè)新函數(shù)包裹了原函數(shù),并且綁定了的指向?yàn)閭魅氲摹? 理解 JavaScript this 文章中已經(jīng)比較全面的分析了 this 在 JavaScript 中的指向問(wèn)題,用一句話來(lái)總結(jié)就是:this 的指向一定是在執(zhí)行時(shí)決定的,...

    duan199226 評(píng)論0 收藏0
  • Brief introduction of how to 'Call, Apply and

    摘要:關(guān)于在絕大多數(shù)情況下,函數(shù)的調(diào)用方式?jīng)Q定了的值。不能在執(zhí)行期間被賦值,并且在每次函數(shù)被調(diào)用時(shí)的值也可能會(huì)不同。它們除了參數(shù)略有不同,其功能完全一樣。它們的第一個(gè)參數(shù)都為將要指向的對(duì)象。 關(guān)于 this 在絕大多數(shù)情況下,函數(shù)的調(diào)用方式?jīng)Q定了this的值。this不能在執(zhí)行期間被賦值,并且在每次函數(shù)被調(diào)用時(shí)this的值也可能會(huì)不同。 全局 this window.something = ...

    incredible 評(píng)論0 收藏0
  • JavaScript的作用域、閉包、(apply, call, bind

    摘要:閉包閉包的概念與詞法域關(guān)系緊密。閉包甚至在函數(shù)已經(jīng)返回后也可以獲取其外部函數(shù)的變量。一種常見(jiàn)的閉包導(dǎo)致的由立即調(diào)用函數(shù)表達(dá)式解決的例子事實(shí)上結(jié)果的所有都是,而不是按順序得出的,。 介紹 JavaScript 有一個(gè)特征————作用域。理解作用域scope可以使你的代碼脫穎而出,減少錯(cuò)誤,幫助你用它構(gòu)造強(qiáng)大的設(shè)計(jì)模式。 什么是作用域 作用域就是在代碼執(zhí)行期間變量,函數(shù)和對(duì)象能被獲取到的特...

    tyheist 評(píng)論0 收藏0
  • Function Definition, This and Bind in JavaScript

    I thought I know the Function definition, execution context and the behavior of this in JavaScript. However, I realized that actually I dont or the knowlege is still not firmly grounded in my mind wh...

    siberiawolf 評(píng)論0 收藏0
  • 不再彷徨:完全弄懂JavaScript中的this(譯文總結(jié))

    摘要:至于,其只是以數(shù)組的方傳入?yún)?shù),其它部分是一樣的,如下它們也可用于在中的類(lèi)繼承中,調(diào)用父級(jí)構(gòu)造器。間接調(diào)用,調(diào)用了父級(jí)構(gòu)造器對(duì)比方法和,它倆都立即執(zhí)行了函數(shù),而函數(shù)返回了一個(gè)新方法,綁定了預(yù)先指定好的,并可以延后調(diào)用。 其實(shí)this是一個(gè)老生常談的問(wèn)題了。關(guān)于this的文章非常多,其實(shí)我本以為自己早弄明白了它,不過(guò)昨天在做項(xiàng)目的過(guò)程中,還是出現(xiàn)了一絲疑惑,想到大概之前在JavaScri...

    YanceyOfficial 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<