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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...