JavaScript中的apply和call函數詳解

来源:http://www.cnblogs.com/sunshine-s/archive/2016/09/25/5906021.html
-Advertisement-
Play Games

本文是翻譯Function.apply and Function.call in JavaScript,希望對大家有所幫助 轉自“http://www.jb51.net/article/52416.htm” 第一次翻譯技術文章,見笑了! 翻譯原文: Function.apply and Functi ...


本文是翻譯Function.apply and Function.call in JavaScript,希望對大家有所幫助

轉自“http://www.jb51.net/article/52416.htm”

第一次翻譯技術文章,見笑了!

翻譯原文:

Function.apply and Function.call in JavaScript

第一段略。

每個JavaScript函數都會有很多附屬的(attached)方法,包括toString()、call()以及apply()。聽起來,你是否會感到奇怪,一個函數可能會有屬於它自己的方法,但是記住,JavaScript中的每個函數都是一個對象。看一下 這篇文章 ,複習一下(refresher)JavaScript特性。你可能還想知道JavaScript中函數和方法的區別。我認為“函數”和“方法”的描述,僅僅是JavaScript的習慣約定而已。函數立足於它們自己(例如:alert()),而方法是函數內部一個對象的屬性(dictionary),我們通過對象來調用方法。每個JavaScript對象都有一個toString()方法,下麵通過代碼舉例說明,在一個函數對象中,我們可以使用toString()方法

1 <script> 
2 function foo(){
3    alert('x');
4 }
5 alert(foo.toString());
6 </script>

 

因為函數都是對象,它們有自己的屬性和方法。我們可以把它們看作數據(data)。這篇文章,我們只關註兩個函數的方法apply()以及call()。

我們從下麵的代碼開始:

1 var x = 10;
2 function f(){
3  alert(this.x);
4 }
5 f();

 

我們定義了一個全局函數f()。f()通過this關鍵字訪問變數x,但是需要註意的是,我們不能通過一個對象的實例來調用這個函數。this指向的是什麼對象呢?this會指向這個全局對象。我們的變數x就是在這個全局對象中定義的。上面的代碼能夠正常運行,運行結果會顯示一個對話框,對話框中顯示10。

我們可以通過this來調用call()和apply()。正如下麵的例子展示如何使用call():

1 var x = 10;
2 var o = { x : 15};
3 function f(){
4  alert(this.x);
5 }
6 f();
7 f.call(o);

首先調用f()將會顯示10的對話框,因為this這個時候指向的是全局對象。然後我們調用f函數的call()方法,傳入的參數是o,運行結果顯示的是o中x屬性的值15。call()方法會用它的第一個參數作為f函數的this指針。也就是說,我們會告訴運行時,f函數中的this指向的是哪個對象。

this跳轉聽起來有些滑稽,甚至對於C++、Java以及C#程式員來說有些反常。這些都是ECMAScript中有趣的部分。

通過call()也可以給函數傳遞參數:

1 var x = 10;
2 var o = { x : 15};
3 function f(){
4  alert(this.x);
5 }
6 f();
7 f.call(o);

 

apply()和call()類似的,只是apply()要求第二個參數必須是一個數組。這個數組會作為參數傳遞給目標函數。

1 var x = 10;
2 var o = {x : 15};
3 function f(message) {
4  alert(message);
5  alert(this.x);
6 }
7 f('invoking f');
8 f.apply(o, ['invoking f through apply']);

apply()方法是很有用的,因為我們可以創建一個函數而不用去關心目標方法的參數。這個函數可以通過apply()的第二個數組參數來傳遞額外的參數給方法。

 1 var o = {x : 15};
 2 function f1(message1) {
 3  alert(message1 + this.x);
 4 }
 5 function f2(message1, message2) {
 6  alert(message1 + (this.x * this.x) + message2);
 7 }
 8 function g(object, func, args) {
 9  func.apply(object, args);
10 }
11 g(o, f1, ['the value of x = ']);
12 g(o, f2, ['the value of x squared = ', '. Wow!']);

這樣的語法有點問題。為了調用apply()方法,我們強制目標函數使用數組中的參數。幸運的是,有一個方法可以讓這種語法更簡單。在此之前,我們必須先介紹一個:參數標識符。

在JavaScript中,其實每個函數都有一個可變長度的參數列表。這意味著,即使一個函數只有一個參數的時候,我們也可以傳遞5個參數給它。下麵的代碼不會有錯誤,而且結果顯示的是“H”。

1 function f(message) {
2  alert(message);
3 }
4 f('H', 'e', 'l', 'l', 'o');

在f()中,如果我們不想去接受其他的參數,我們可以用關鍵字arguments。arguments代表一個參數對象,它有一個代表長度的屬性類似於數組。

1 function f(message) {
2  // message的值和arguments[0]是一樣的
3  for(var i = 1; i < arguments.length; i++){
4   message += arguments[i];
5  }
6  alert(message);
7 }
8 // 結果顯示“Hello”
9 f('H', 'e', 'l', 'l', 'o');

你應該知道,嚴格來講,arguments不是一個數組。arguments有一個length屬性,但是沒有split、push、pop方法。在前面的g()函數中,我們可以從arguments中拷貝需要的參數,組成數組,然後把這個數組傳遞給apply()。

 1 var o = {x : 15};
 2 function f(message1, message2) {
 3  alert(message1 + ( this.x * this.x) + message2);
 4 }
 5 function g(object, func) {
 6  // arguments[0] = object
 7  // arguments[1] = func
 8  var args = [];
 9  for(var i = 2; i < arguments.length; i++) {
10   args.push(arguments[i]);
11  }
12  func.apply(object, args);
13 }
14 g(o, f, 'The value of x squared = ', '. Wow!');

 

當我們調用g(),we can pass additional arguments as parameters instead of stuffing the arguments into an array。


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

-Advertisement-
Play Games
更多相關文章
  • 首先看兩段很相似的代碼: 1. 2. 兩個例子都是重寫了原型對象,但是實例創建的順序直接導致了倆個輸出的結果。 第一個彈出 “Nic”,而第二個會報錯“TypeError: friend2.sayName is not a function” !!! 兩端代碼唯一的區別就是實例在原型對象重寫之前還是 ...
  • 微信小程式全稱微信公眾平臺·小程式,原名微信公眾平臺·應用號(簡稱微信應用號) 聲明 微信小程式開發工具類似於一個輕量級的IDE集成開發環境,目前僅開放給了少部分受微信官方邀請的人士(據說僅200個名額)進行內測,因此目前未受到邀請的人士只能使用破解版; 本破解版資源來自於網上,與本人無關,僅供技術 ...
  • function clickFunc(id) { var params = new Array(); params.push({ name: "id", value: id}); Post("/Action/method", params);}function Post(URL, PARAMTERS ...
  • 20:32:59 Ctrl+/ 或 Ctrl+Shift+/ 註釋(// 或者 ) Shift+F6 重構-重命名 Ctrl+X 刪除行 Ctrl+D 複製行 Ctrl+G 查找行 Ctrl+Shift+Up/Down 代碼向上/下移動。 F2 或Shift+F2 高亮錯誤或警告快速定位 寫代碼,按 ...
  • 事件背景: 最近在做一個移動端業務的時候碰到一個bug,在ios上對body綁定click事實現事件代理冒泡至某些元素上盡然不生效。 思考: 暫藉助jquery展示下事件綁定代碼,將所有標簽含有data-tip屬性的元素通過事件代理至body 這樣做在android和pc上都可以正常實現,但是在io ...
  • js中三種彈窗 1)alert 彈出警告 無返回值---------alert('第一行\n第二行'); 2)confirm()選擇確定或取消,返回t或f----var result = confirm('是否刪除!'); 3)prompt()彈出輸入框,返回輸入內容----var value = ...
  • 以下是個人對這三個老大難的總結(最近一直在學習原生JS,翻了不少書,不少文檔,雖然還是新手,但我會繼續堅持走我自己的路) 原型鏈 所有對象都是基於 ,`Object.prototype Object.prototype Object.prototype toString() [[class]] '[ ...
  • <!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1,maximum-scale=1, ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...