JavaScript中的apply()、call()、bind()

来源:https://www.cnblogs.com/uakora/archive/2020/04/12/12684529.html
-Advertisement-
Play Games

JavaScript 中 apply、call、bind方法的異同: 相同點 都是用來動態指定函數 this 對象的指向 第一個參數都是 this 要指向的對象,也就是要指定的上下文 都可以利用後續參數傳參 不同點 傳參形式不同:apply 方法接受的是一個參數數組,call 和 bind 方法接受 ...


JavaScript 中 apply、call、bind方法的異同:

相同點

  • 都是用來動態指定函數 this 對象的指向
  • 第一個參數都是 this 要指向的對象,也就是要指定的上下文
  • 都可以利用後續參數傳參

不同點

  • 傳參形式不同:apply 方法接受的是一個參數數組,call 和 bind 方法接受的是參數列表
  • 執行方式不同:apply、call 會立即執行,而 bind 方法會創建一個新函數,需要單獨調用執行

apply() 的使用

語法:func.apply(thisArg, [argsArray])

thisArg: 必選的。在 func 函數運行時使用的 this 值

argsArray: 可選的。一個數組或者類數組對象,其中的數組元素將作為單獨的參數傳給 func 函數

var name = 'globalName';
var age = 'globalAge';

var person = {
  name: '姓名',
  age: '年齡',
  foo: function() {
    console.log('姓名:' + this.name + ' ' + '年齡:' + this.age);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                   // 姓名:姓名 年齡:年齡
person.foo.apply(xiaoming);     // 姓名:小明 年齡:25
person.foo.apply(null);         // 姓名:globalName 年齡:globalAge (非嚴格模式)
person.foo.apply(undefined);    // 姓名:globalName 年齡:globalAge (非嚴格模式)
person.foo.apply({});           // 姓名:undefined 年齡:undefined

var foo = person.foo;
foo();                          // 姓名:globalName 年齡:globalAge (非嚴格模式)
foo.apply(person);              // 姓名:姓名 年齡:年齡
foo.apply(xiaoming);            // 姓名:小明 年齡:25
/* 傳參: 一個數組 */
var person = {
  name: '姓名',
  age: '年齡',
  foo: function(a, b, c) {
    console.log(a, b, c);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                              // undefined undefined undefined
person.foo.apply(xiaoming, [1, 2, 3]);     // 1 2 3

var foo = person.foo;
foo.apply(xiaoming, [1, 2, 3]);            // 1 2 3

call() 的使用

語法:function.call(thisArg, arg1, arg2, ...)

thisArg: 可選的。在 function 函數運行時使用的 this 值

arg1, arg2, ...: 可選的。指定的參數列表

var name = 'globalName';
var age = 'globalAge';

var person = {
  name: '姓名',
  age: '年齡',
  foo: function() {
    console.log(this.name + ' ' + this.age);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                  // 姓名 年齡
person.foo.call(xiaoming);     // 小明 25
person.foo.call(null);         // globalName globalAge (非嚴格模式)
person.foo.call(undefined);    // globalName globalAge (非嚴格模式)
person.foo.call({});           // undefined undefined

var foo = person.foo;
foo();                         // globalName globalAge (非嚴格模式)
foo.call(person);              // 姓名 年齡
foo.call(xiaoming);            // 小明 25

說明:不傳參,用法類似 apply()

/* 傳參: 列表 */
var person = {
  name: '姓名',
  age: '年齡',
  foo: function(a, b, c) {
    console.log(a, b, c);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                            // undefined undefined undefined
person.foo.call(xiaoming, 1, 2, 3);     // 1 2 3

var foo = person.foo;
foo.call(xiaoming, 1, 2, 3);            // 1 2 3

bind() 的使用

語法: function.bind(thisArg[, arg1[, arg2[, ...]]])

參數
thisArg: 調用綁定函數時作為 this 參數傳遞給目標函數的值。 如果使用new運算符構造綁定函數,則忽略該值。

arg1, arg2, ...: 當目標函數被調用時,被預置入綁定函數的參數列表中的參數

返回值
返回一個原函數的拷貝,並擁有指定的 this 值和初始參數

var name = 'globalName';
var age = 'globalAge';

var person = {
  name: '姓名',
  age: '年齡',
  foo: function() {
    console.log(this.name + ' ' + this.age);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                    // 姓名 年齡
person.foo.bind(xiaoming)();     // 小明 25
person.foo.bind(null)();         // globalName globalAge (非嚴格模式)
person.foo.bind(undefined)();    // globalName globalAge (非嚴格模式)
person.foo.bind({})();           // undefined undefined

var foo = person.foo;
foo();                           // globalName globalAge (非嚴格模式)
foo.bind(person)();              // 姓名 年齡

var bar = foo.bind(xiaoming);            
bar();                           // 小明 25 (需要單獨調用執行)

說明:相比 apply()、call()會立即執行,bind 需要單獨調用執行 bind()()

/* 傳參: 列表 */
var person = {
  name: '姓名',
  age: '年齡',
  foo: function(a, b, c) {
    console.log(a, b, c);
  }
}
var xiaoming = {
  name: '小明',
  age: 25
};

person.foo();                             // undefined undefined undefined
person.foo.bind(xiaoming, 1, 2, 3)();     // 1 2 3

var foo = person.foo;
var bar = foo.bind(xiaoming, 1);          // 返回原函數的拷貝         
bar(2, 3);                                // 1 2 3                        


以上只是 apply()、call()、bind() 方法主要的一個用法,更多用法詳細請閱讀MDN相關文檔


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

-Advertisement-
Play Games
更多相關文章
  • Web 開發中幾乎的平臺都需要一個後臺管理,但是從零開發一套後臺控制面板並不容易,幸運的是有很多開源免費的後臺控制面板可以給開發者使用。 ...
  • 為什麼不切圖標了 以前的圖標我們非常喜歡用ps等工具切成一張張xxx.png圖片,如果稍微懂點移動端適配,對dpr(設備像素比)有所瞭解的,還會切出類似[email protected],[email protected],[email protected]這樣的圖標,其中@1x表示用來適配dpr為1的手機,圖像解析度最低;@2x表 ...
  • 需求 ​ 通過JavaScript使盒子跟隨著滑鼠的移動而移動,並且當頁面出現滾動條時,盒子也會跟隨滑鼠正常移動(相容IE8) ​ 代碼 ...
  • 什麼是 JSX 語法 JSX語法里,有兩種類型的標簽: 1、普通的html標簽(首字母小寫) 2、組件標簽(首字母大寫) 使用 React 編寫 TodoList 功能 src/TodoList.js import React,{Fragment} from 'react'; function To ...
  • 數組是用於儲存多個相同類型數據的集合,JavaScript 開發中數組開發是必須掌握技能,工作學習中沒少和數組打交道,所以重中之重必須掌握,以下是數組中常用方法及實例。 數組常用操作 1、創建數組 直接定義數組項方法; 構造函數new Array('Apple', 'Banana'); 定義一個空數 ...
  • WordPress通知欄可有效地將流量引至您期望的頁面。此過程可以增加您網站的收入。通知欄可以顯示有關您網站的重要公告,也可以使用該欄捕獲訪問者的電子郵件以增加訂閱。這些欄對促進銷售特別有用,這樣用戶就不會錯過以折扣價購買產品和服務的機會,從而幫助您增加收入。現代的通知欄不僅功能強大,而且功能強大。 ...
  • src/Test.js import React from 'react'; function Test() { return ( <div> test </div> ); } export default Test; src/index.js中引入組件Test 頁面中效果 ...
  • react fiber 指react 16以上的版本 引入react的方式: 1、引入.js文件 2、使用腳手架工具(推薦) 推薦使用react官方提供的腳手架工具:create-react-app React開發環境準備 (npx 是 npm 的高級版本,npx 具有更強大的功能) npx cre ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...