js基礎篇——call/apply、arguments、undefined/null

来源:http://www.cnblogs.com/chuaWeb/archive/2016/02/15/5190770.html
-Advertisement-
Play Games

a.call和apply方法詳解 call方法: 語法:call([thisObj[,arg1[, arg2[, [,.argN]]]]]) 定義:調用一個對象的一個方法,以另一個對象替換當前對象。 說明: call 方法可以用來代替另一個對象調用一個方法。call 方法可將一個函數的對象上下文從初


a.call和apply方法詳解


call方法: 

  語法:call([thisObj[,arg1[, arg2[,   [,.argN]]]]]) 

  定義:調用一個對象的一個方法,以另一個對象替換當前對象。 

  說明: call 方法可以用來代替另一個對象調用一個方法。call 方法可將一個函數的對象上下文從初始的上下文改變為由 thisObj 指定的新對象。如果沒有提供 thisObj 參數,那麼 Global 對象被用作 thisObj。 

 

apply方法: 

  語法:apply([thisObj[,argArray]]) 

  定義:應用某一對象的一個方法,用另一個對象替換當前對象。

  說明:如果 argArray 不是一個有效的數組或者不是 arguments 對象,那麼將導致一個 TypeError。如果沒有提供 argArray 和 thisObj 任何一個參數,那麼 Global 對象將被用作 thisObj, 並且無法被傳遞任何參數。

 

實例學習:

function add(a,b){ alert(a+b);}
function sub(a,b){ alert(a-b);}
add.call(sub,3,1);

  列印結果為4。調用add函數,但是調用對象(上下文環境)不是add對象,而是sub函數對象。註意:js中的函數其實是對象,函數名是對 Function 對象的引用。

 

function Animal(){ 
this.name = "Animal"; 
this.showName = function(){  alert(this.name);} 
}
function Cat(){  this.name = "Cat"; } 
var animal = new Animal(); 
var cat = new Cat(); 
animal.showName.call(cat,",");//輸出結果為"Cat"
animal.showName.apply(cat,[]);//輸出結果為"Cat"

  call 的意思是把 animal 的方法放到cat上執行,上下文環境為cat,原來cat是沒有showName() 方法,現在是把animal 的showName()方法放到 cat上來執行,而cat的this.name是Cat。所以this.name 應該是 Cat

 

實現繼承

function Animal(name){   
this.name = name;   
this.showName = function(){  alert(this.name);}   
}   
function Cat(name){  Animal.call(this, name); }   
var cat = new Cat("Black Cat");  
cat.showName();

  Animal.call(this) 的意思就是調用Animal方法,但是使用 this對象代替Animal對象,上下文環境變成了this。new Cat("Black Cat")中使用Animal.call給當前的上下文環境設置了屬性name和方法showName。

         拓展:多重繼承

function Class10(){
this.showSub = function(a,b){  alert(a-b); }
}

function Class11(){
this.showAdd = function(a,b){   alert(a+b);    }
}

function Class2(){
    Class10.call(this);
    Class11.call(this);
}

  備註:js的繼承還有其他方法,例如使用原型鏈,這個不屬於本文的範疇,只是在此說明call 的用法。說了call ,當然還有 apply,這兩個方法基本上是一個意思,區別在於 call 的第二個參數可以是任意類型,而apply的第二個參數必須是數組或arguments。

 

b.arguments使用


什麼是arguments

  arguments 是是JavaScript里的一個內置對象,它很古怪,也經常被人所忽視,但實際上是很重要的。所有主要的js函數庫都利用了arguments對象。所以agruments對象對於javascript程式員來說是必需熟悉的。

  所有的函數都有屬於自己的一個arguments對象,它包括了函所要調用的參數。他不是一個數組,如果用typeof arguments,返回的是’object’。雖然我們可以用調用數據的方法來調用arguments。比如length,還有index方法。但是數 組的push和pop對象是不適用的。

 

使用arguments創建一個靈活的函數

  看起來貌似argument對象使用起來十分有限,但是實際上它是一個非常有用的對象。你可以通過使用argument對象讓函數能夠調用數量不定 的參數。在Dean Edwards的base2庫里有個格式化的函數,展示了這個靈活性。

function format(string) {
var args = arguments;
var pattern = new RegExp('%([1-' + arguments.length + '])', 'g');
return String(string).replace(pattern, function(match, index,position,all) {  
        console.log(match + '&' + index + '&' + position + '&' + all);
return args[index];  
    });  
};

  掉用format('And the %1 want to know whose %2 you %3', 'papers', 'shirt', 'wear');結果為"And the papers want to know whose shirt you wear";控制台列印為

  %1&1&8&And the %1 want to know whose %2 you %3

  %2&2&30&And the %1 want to know whose %2 you %3

  %3&3&37&And the %1 want to know whose %2 you %3

 

把arguments對象轉換成一個真正的數組

  雖然arguments對象不是一個真正的javascript數組,但是我們還是可以輕易的把它轉換成標準的數據 ,然後進行數組操作。

  var args = Array.prototype.slice.call(arguments); 

  那麼現在這個變數args就含有一個含有函數所有參數的標準javascript數組對象。

        

拓展:使用上一節的format函數,通過預置的arguments對象創建函數

 function makeFunc() {   
var args = Array.prototype.slice.call(arguments);   
var func = args.shift();   
return function() {   
return func.apply(null, args.concat(Array.prototype.slice.call(arguments)));   
    };   
}

  該方法會將第一個參數取出來,然後返回一個curry化函數,該curry化函數的參數(第二個arguments)將和makeFunc的從第二個參數開始的參數組合成新數組。並返回makeFunc第一個參數的apply調用

  執行

var majorTom = makeFunc(format, "This is Major Tom to ground control. I’m %1.");
majorTom("stepping through the door");

  結果為:"This is Major Tom to ground control. I’m stepping through the door."

  控制台列印:%1&1&41&This is Major Tom to ground control. I’m %1.

 

 [function.]arguments.callee

  說明:arguments.callee方法返回的是正在執行的函數本身。

  callee 屬性是 arguments 對象的一個成員,它表示對函數對象本身的引用,這有利於匿名函數的遞歸或者保證函數的封裝性,例如下邊示例的遞歸計算1到n的自然數之和。而該屬性僅當相關函數正在執行時才可用。還有需要註意的是callee擁有length屬性,這個屬性有時候用於驗證還是比較好的。arguments.length是實參長度arguments.callee.length是形參(定義時規定的需要的參數)長度,由此可以判斷調用時形參長度是否和實參長度一致。

//用於驗證參數
function calleeLengthDemo(arg1, arg2) {
if (arguments.length==arguments.callee.length) {
    window.alert("驗證形參和實參長度正確!");
return;
  } else {
    alert("實參長度:" +arguments.length);
    alert("形參長度: " +arguments.callee.length);
  }
}

//遞歸計算
var sum = function(n){
if (n <= 0) return 1;
else return n +arguments.callee(n - 1)
}

//比較一般的遞歸函數:
var sum = function(n){
if (1==n) return 1;
else return n + sum (n-1);
}

  調用時:alert(sum(100));其中函數內部包含了對sum自身的引用,函數名僅僅是一個變數名,在函數內部調用sum即相當於調用一個全局變數,不能很好的體現出是調用自身,這時使用callee會是一個比較好的方法。

 

拓展 functionName.caller 

  說明: 返回是誰調用了functionName 函數。functionName 對象是所執行函數的名稱。對於函數來說,caller 屬性只有在函數執行時才有定義。如果函數是由頂層調用的,那麼 caller 包含的就是 null 。如果在字元串上下文中使用 caller 屬性,那麼結果和 functionName.toString 一樣,也就是說,顯示的是函數的反編譯文本。 下麵的例子說明瞭 caller 屬性的用法:

// caller demo {
function callerDemo() {
if (callerDemo.caller) {
var a= callerDemo.caller.toString();
    alert(a);
  } else {
    alert("this is a top function");
  }
}

function handleCaller() {
  callerDemo();
}

handleCaller();

  執行結果:

  

  

 c.undefined和null


  大多數電腦語言,有且僅有一個表示"無"的值,比如,C語言的NULL,Java語言的null,Python語言的none,Ruby語言的nil。有點奇怪的是,JavaScript語言居然有兩個表示"無"的值:undefined和null。這是為什麼?

相似性
  在JavaScript中,將一個變數賦值為undefined或null,老實說,幾乎沒區別。

  代碼如下:

var a = undefined;
var a = null;

  上面代碼中,a變數分別被賦值為undefined和null,這兩種寫法幾乎等價。

  undefined和null在if語句中,都會被自動轉為false,相等運算符甚至直接報告兩者相等。

if (!undefined) 
    console.log('undefined is false');
// undefined is false

if (!null) 
    console.log('null is false');
// null is false

undefined == null
// true

  上面代碼說明,兩者的行為是何等相似!但是我們去查看undefined和null的各自的類型卻發現類型是不同的。js基礎類型中沒有null類型

typeof null;//"object"
typeof undefined;//"undefined"

  既然undefined和null的含義與用法都差不多,為什麼要同時設置兩個這樣的值,這不是無端增加JavaScript的複雜度,令初學者困擾嗎?Google公司開發的JavaScript語言的替代品Dart語言,就明確規定只有null,沒有undefined!

 

歷史原因

  原來,這與JavaScript的歷史有關。1995年JavaScript誕生時,最初像Java一樣,只設置了null作為表示"無"的值。

  根據C語言的傳統,null被設計成可以自動轉為0。

Number(null) // 0

5 + null // 5

  但是,JavaScript的設計者Brendan Eich,覺得這樣做還不夠,有兩個原因。

  首先,null像在Java里一樣,被當成一個對象。

typeof null // "object"

  但是,JavaScript的數據類型分成原始類型(primitive)和合成類型(complex)兩大類,Brendan Eich覺得表示"無"的值最好不是對象。

  其次,JavaScript的最初版本沒有包括錯誤處理機制,發生數據類型不匹配時,往往是自動轉換類型或者默默地失敗。Brendan Eich覺得,如果null自動轉為0,很不容易發現錯誤。因此,Brendan Eich又設計了一個undefined。

 

最初設計

  JavaScript的最初版本是這樣區分的:null是一個表示"無"的對象,轉為數值時為0;undefined是一個表示"無"的原始值,轉為數值時為NaN。

Number(undefined)  // NaN
5 + undefined  // NaN

 

目前的用法

  但是,上面這樣的區分,在實踐中很快就被證明不可行。目前,null和undefined基本是同義的,只有一些細微的差別。

  null表示"沒有對象",即該處不應該有值。典型用法是:

  (1) 作為函數的參數,表示該函數的參數不是對象。

  (2) 作為對象原型鏈的終點。

Object.getPrototypeOf(Object.prototype)  // null

  undefined表示"缺少值",就是此處應該有一個值,但是還沒有定義。典型用法是:

  (1)變數被聲明瞭,但沒有賦值時,就等於undefined。

  (2) 調用函數時,應該提供的參數沒有提供,該參數等於undefined。

  (3)對象沒有賦值的屬性,該屬性的值為undefined。

  (4)函數沒有返回值時,預設返回undefined。

var i;
i // undefined

function f(x){console.log(x)}
f() // undefined

var  o = new Object();
o.p // undefined

var x = f();
x // undefined

 

  上面的文章部分是摘自網上的文章記錄下的古老筆記,記錄的時候沒有註明出處現在也不好找了,如果某大牛路過發現裡面包含您寫的博文,請在評論中註明出處,本人補上,謝謝。

  

  如果覺得本文不錯,請點擊右下方【推薦】!


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

-Advertisement-
Play Games
更多相關文章
  • js如何合併兩個數組:本章節介紹一下如何合併兩個數組,希望對大家有所幫助。實例代碼:實例一: var arrOne=["螞蟻部落","青島市南區",3]; var arrTwo=["分享交流","antzone",600]; console.log(arrOne.concat(arrTwo)); c
  • css3實現元素的伸縮效果代碼實例:本章節分享一段代碼實例,它利用css3實現了元素的伸縮效果。這裡為了進行演示,結合了javascript代碼。代碼實例如下: <!DOCTYPE html> <html> <head> <meta charset=" utf-8"> <meta name="aut
  • jquery實現的點擊可以展開摺疊的垂直導航菜單:本章節介紹一種比較常見的導航菜單是如何實現的,它具有垂直結構,點擊導航主標題可以展開或者摺疊二級菜單。代碼實例如下: <!DOCTYPE html> <html> <head> <meta charset=" utf-8"> <meta name="
  • js獲取地址欄的參數:通過url地址可以傳遞參數,下麵就是一段能夠實現此功能的代碼實例和和大家做一下分享,希望能夠給需要的朋友帶來一定的幫助。代碼如下: function getUrlParams(){ var search = window.location.search; // 寫入數據字典 v
  • 效果預覽:http://hovertree.com/code/texiao/ks63r6aq.htm 1 <!DOCTYPE html> 2 <html xmlns="http://www.w3.org/1999/xhtml"> 3 <head> 4 <title>超漂亮的HTML導航菜單CSS代碼
  • 背景: 刷知乎時,老是需要用滑鼠點下一頁,對於懶人來說,太過麻煩。 功能: 敲擊 “←” 、“→”,實現上一頁、下一頁功能。 思路: 通過谷歌擴展來實現:監聽鍵盤事件,如果keycode為37、39時,進行上一頁、下一頁操作。 疑問: 本想著如果檢測到keycode為37、39時,直接觸發按鈕“上一
  • 今天跟大家分享的是一個依賴於angular的上傳控制項。 前段時間做項目遇到一個需求是上傳文件,大概需要實現的樣式是這樣子的,見下圖: 需要同時上傳兩個文件。並且規定文件格式和文件大小。因為前端框架使用angular,且不想因為一個上傳功能又引入一個jquery,所以在網上查找基於angular的上傳
  • 一說起CSS隱藏元素,我想大部分小伙伴們都會想到的第一種方法就是設置display為none。這是最為人所熟知也是最常用的方法。我相信還有不少人想到使用設置visibility為hidden來隱藏元素,這種方式也是常用的方法,而且也有很多人知道兩者的不同。除了這兩種方法,本文還總結了一些比較不常用的...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...