解析JavaScript中apply和call以及bind

来源:http://www.cnblogs.com/androidshouce/archive/2016/06/15/5586191.html
-Advertisement-
Play Games

函數調用方法 在談論JavaScript中apply、call和bind這三兄弟之前,我想先說下,函數的調用方式有哪些: 作為函數 作為方法 作為構造函數 通過它們的call()和apply()方法間接調用 前面的三種調用方法,我們都知道且不在這篇文章的討論範圍內,就不說了。 下麵我們來說說這第四種 ...


函數調用方法

在談論JavaScript中apply、call和bind這三兄弟之前,我想先說下,函數的調用方式有哪些:

  • 作為函數
  • 作為方法
  • 作為構造函數
  • 通過它們的call()和apply()方法間接調用

前面的三種調用方法,我們都知道且不在這篇文章的討論範圍內,就不說了。

下麵我們來說說這第四種調用方法

通過call()和apply()間接調用

其實,我們可以將這兩個函數看做是某個對象的方法,通過調用方法的方式來間接調用函數:

function f(){}

f.call(o);
f.apply(o);

call()和apply()的第一個參數是要調用函數的母對象,它是調用上下文,在函數體內通過this來獲得對它的引用。

那麼他們是一樣的,還是有區別的,還有bind方法呢?君莫急,下麵詳細的解析他們三者的區別和聯繫。

call()

call()方法給調用它的方法指定特定的this指針(習慣用語,不要和我糾結它的正確性)和參數。例如有這麼一個函數:

var fn = function (arg1, arg2) {
  console.log(this, arg1, arg2);      
}

我來調用它:

fn.call(null, 'Skylor', 'min');        //1
fn.call(undefined, 'Skylor', 'min');   //2

var fx = function() {}

fn.call(fx, 'Skylor', 'min');          //3

這三個call方法的返回值是什麼呢?不廢話,請看:

1.  null "Skylor" "min"
2.  undefined "Skylor" "min"
3.  fx "Skylor" "min"

真的是這樣嗎,機智的你,去瀏覽器控制台小試了一下,我去,你這坑貨,不是這樣的:

chrome
1.  Window "Skylor" "min"
2.  Window "Skylor" "min"
3.  fx "Skylor" "min"

好吧,你機智。但這已經很好的說明瞭call方法了。(window很高級,微軟偷笑中...)

我們註意到call方法,第一個參數是指定this指針,後面每個參數指定需要的參數,註意我用的是“每個”,這意味著你需要幾個參數就要想調用函數那樣,一個個參數寫進去。

apply()

apply()是call()的兄弟啊,其他地方長的都一樣,都是男的,就一個地方不一樣。先看例子:

fn.apply(null, ['Skylor', 'min']); //1
fn.apply(fx, ['Skylor', 'min']);   //2

哥們,你是不是寫錯了,多了個中括弧啊。不,不,不,這就是他和call長的不一樣的地方,它的第二個參數是個需要的參數Array。

bind()

bind()嘛,他們三個不是仨兄弟嘛,,這個我懂,blabla....不不不,它是和apply、call結拜的兄弟,不是親兄弟。

當然,bind方法也是允許你指定this指針,但是它不是調用函數,而是返回一個(或者說是拷貝調用它的函數的函數,並給這個函數指定特定的this指針和參數。慣例,例子說明一切:

var fnbound = fn.bind(null, 'Skylor', 'min');

這時,fnbound是一個函數,一個this指向null,參數為['Skylor', 'min']的另一個函數。調用之:

fnbound();

結果:

null, 'Skylor', 'min'

不要和我糾結Window的事了。。。。。

bind和其他兩個兄弟不一樣的地方,是,它不是調用函數,而是返回一個新的函數,同樣,它也是指定this指針和參數的,指定參數的方式和call一樣,是一個一個來的。

最後來一個例子唄:

複製代碼
var shoppingCart = (function(){
         var _calculatePrice = function () {
                    return this.price * this.amount;
         };

         return {
                    calculatePrice : _calculatePrice
         }
})();


var goods = {
           name : ‘hammer’,
           price: 199,
           amount : 2
};

shoppingCart.calculatePrice.call(goods);
複製代碼

 

OK,以上是我的見解,如有不正確的地方,歡迎指正和批評,不甚歡喜與感謝。By Skylormin

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 註冊工廠是一種很常用的框架書寫方法,它適合於快速創建相同類型的對象。 舉個慄子 比如一個傢具工廠,有沙發、椅子、茶几等等,正常的編程模式是這樣的: 如果想要擴展,就需要繼續定義class,然後new對象。 但是其實沙發的製作與使用時解耦的,使用者並不需要知道沙發、茶几是怎麼製作出來的,只是想使用它而 ...
  • 繼承時實現代碼重用的重要手段,但它並非永遠是完成這項工作的最佳工具,不恰當的使用會導致程式變得很脆弱,當然,在同一個程式員的控制下,使用繼承會變的非常安全。想到了很有名的一句話,你永遠不知道你的用戶是如何使用你寫的程式的,一個程式員繼承另一個程式員寫的類也是同樣的危險。 於方法調用不同的是,繼承打破 ...
  • 引言:正則表達式為高級的文本模式匹配、抽取、與/或文本形式和替換功能提供基礎。在python中,通過標準庫中的re模塊來支持正則表達式 '.'點號,在普通模式,它匹配除換行符外的任意一個字元;如果指定了 DOTALL 標記,匹配包括換行符以內的任意一個字元。 '^'尖尖號,匹配一個字元串的開始,在 ...
  • 安裝 要在PHP程式中使用Redis,首先需要確保 Redis 的PHP驅動程式和 PHP 安裝設置在機器上。可以查看 PHP教程 教你如何在機器上安裝PHP。現在,讓我們來看看一下如何設置 Redis 的PHP驅動程式。 需要從 github 上資料庫: https://github.com/ni ...
  • 一、Redis基礎部分: 1、redis介紹與安裝比mysql快10倍以上 *****************redis適用場合**************** 1.取最新N個數據的操作 2.排行榜應用,取TOP N 操作 3.需要精確設定過期時間的應用 4.計數器應用 5.Uniq操作,獲取某段時 ...
  • python新手經常會犯的錯誤小結,有匿名函數lambda的理解,一個元素的元組,模塊導入,參數傳遞是可變的還是不可變的,迭代器和生成器的理解 ...
  • 當需求中是體現部分與整體層次的結構時,以及你希望忽略組合對象與單個對象的不同,統一的使用組合結構中的所有對象時,就應該考慮使用組合模式了。例如:我們單個複製一個文件和多個複製文件,對我們而言,並不在乎一個文件的複製與多個文件複製的區別,也就是我們的操作是一樣的。 下麵的代碼是建立一個公司的組織結構, ...
  • 1.先奉上整理的14圖。 2.其次奉上整理的圖之間的6種關係 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...