JavaScript類型檢測之typeof 和 instanceof 的缺陷與最佳化,typeofinstanceof

來源:互聯網
上載者:User

JavaScript類型檢測之typeof 和 instanceof 的缺陷與最佳化,typeofinstanceof

在javascript中,typeof 和 instanceof 是用來判斷資料類型比較通用的兩個方法,這篇文章的目的是通過對這兩個方法介紹來分析其存在的不足並提出最佳化方案。

typeof

--------------------------------------------------------------------------------

typeof 返回一個運算式的資料類型的字串,返回結果為javascript中的基礎資料型別 (Elementary Data Type),包括:number、boolean、string、object、undefined、function等6種資料類型。

typeof 100; //numbertypeof (1==1); //booleantypeof 'onepixel'; //stringtypeof {} ; //objecttypeof onepixel; // undefinedtypeof parseInt; // functiontypeof [];//objecttypeof new Date(); //object 

可以看出,typeof 可以準確的判斷除object以外的基礎資料類型,但不能區分object類型的具體類型,比如 Array 、Date 以及自訂類。

instanceof

--------------------------------------------------------------------------------

instanceof 本意是用來判斷 A 是否為 B 的執行個體對象,運算式為:A instanceof B,如果A是B的執行個體,則返回true,否則返回false。 在這裡需要特別注意的是:instanceof檢測的是原型,那它是怎麼檢測的呢,我們用一段虛擬碼來類比其內部執行過程:

instanceof (A,B) = {var L = A.__proto__;var R = B.prototype;if(L === R) {//A的內部屬性__proto__指向B的原型對象return true;}return false;} 

從上述過程可以看出,當A的__proto__ 指向B的prototype時,就認為A就是B的執行個體對象,我們再來看幾個例子:

[] instanceof Array; //true{} instanceof Object;//truenew Date() instanceof Date;//truefunction Person(){};new Person() instanceof Person;[] instanceof Object; //truenew Date() instanceof Object;//trunew Person instanceof Object;//true 

從上面的例子中,我們發現雖然instanceof能夠正確判斷[] 是Array的執行個體對象,但不能辨別 [] 不是Object的執行個體對象,為什麼呢,這還需要從javascript的原型鏈說起,我們首先來分析一下[]、Array、Object 三者之間的關係,從instanceof判斷能夠得出:[].__proto__ ->Array.prototype, 而Array.prototype.__proto__指向了Object.prototype,Object.prototype.__proto__ 指向了null,標誌著原型鏈的結束。(ps:關於JS原型鏈請閱讀:淺談javascript原型和原型鏈) 因此,[]、Array、Object就形成了一條原型鏈:

從原型鏈可以看出,[]的__proto__最終指向了Object.prototype,類似的new Date()、new Person() 也會形成這樣一條原型鏈,因此,我們用 instanceof 也不能完全精確的判斷object類的具體資料類型。

最佳化方案

--------------------------------------------------------------------------------

對於這個問題,在閱讀jQuery源碼時,發現了一個比較好的解決方案,由於源碼之間存在相互調用不便於閱讀和理解,因此,按照其思路進行了整理和封裝,代碼如下:

(function(){var class2type = {};var typeList = "Boolean Number String Function Array Date RegExp Object Error Symbol".split( " " );typeList.eachEach(function(item){class2type[ "[object " + item + "]" ] = item.toLowerCase();}return {getObjType:function(obj) {if ( obj == null ) {return obj + "";}if(typeof obj === "object" || typeof obj === "function"){class2type[ toString.call( obj ) ] || "object"}else {return typeof obj;}}}})()

JavaScript 中 typeof 和 instanceof 常用來判斷一個變數是否為空白,或者是什麼類型的。但它們之間還是有區別的:

typeof

typeof 是一個一元運算,放在一個運算數之前,運算數可以是任意類型。

它傳回值是一個字串,該字串說明運算數的類型。typeof 一般只能返回如下幾個結果:

number,boolean,string,function,object,undefined。我們可以使用 typeof 來擷取一個變數是否存在,如 if(typeof a!="undefined"){alert("ok")},而不要去使用 if(a) 因為如果 a 不存在(未聲明)則會出錯,對於 Array,Null 等特殊對象使用 typeof 一律返回 object,這正是 typeof 的局限性。

網上的一個小例子:

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="http://www.w3.org/1999/xhtml"><head><meta http-equiv="Content-Type" content="text/html; charset=utf-8" /><script language="javascript" type="text/javascript">document.write ("typeof(1): "+typeof(1)+"<br>");document.write ("typeof(NaN): "+typeof(NaN)+"<br>");document.write ("typeof(Number.MIN_VALUE): "+typeof(Number.MIN_VALUE)+"<br>");document.write ("typeof(Infinity): "+typeof(Infinity)+"<br>");document.write ("typeof(\"123\"): "+typeof("123")+"<br>");document.write ("typeof(true): "+typeof(true)+"<br>");document.write ("typeof(window): "+typeof(window)+"<br>");document.write ("typeof(Array()): "+typeof(new Array())+"<br>");document.write ("typeof(function(){}): "+typeof(function(){})+"<br>");document.write ("typeof(document): "+typeof(document)+"<br>");document.write ("typeof(null): "+typeof(null)+"<br>");document.write ("typeof(eval): "+typeof(eval)+"<br>");document.write ("typeof(Date): "+typeof(Date)+"<br>");document.write ("typeof(sss): "+typeof(sss)+"<br>");document.write ("typeof(undefined): "+typeof(undefined)+"<br>")</script><title>javascript類型測試</title></head><body></body></html>

instanceof

instance:執行個體,例子

a instanceof b?alert("true"):alert("false"); //a是b的執行個體?真:假

instanceof 用於判斷一個變數是否某個對象的執行個體,如 var a=new Array();alert(a instanceof Array); 會返回 true,同時 alert(a instanceof Object) 也會返回 true;這是因為 Array 是 object 的子類。再如:function test(){};var a=new test();alert(a instanceof test) 會返回
談到 instanceof 我們要多插入一個問題,就是 function 的 arguments,我們大家也許都認為 arguments 是一個 Array,但如果使用 instaceof 去測試會發現 arguments 不是一個 Array 對象,儘管看起來很像。

另外:

測試 var a=new Array();if (a instanceof Object) alert('Y');else alert('N');

得'Y'

但 if (window instanceof Object) alert('Y');else alert('N');

得'N'

所以,這裡的 instanceof 測試的 object 是指 js 文法中的 object,不是指 dom 模型對象。

使用 typeof 會有些區別

alert(typeof(window)) 會得 object

大家知道JavaScript中判斷函數參數類型是用typeof還是instanceof嗎?

typeof只能判斷js已有的幾個類型,如function,object,number。

而instanceof可以判斷對象是由哪個函數執行個體化出來的,如:

var a=function(x){};var b=function(x){};var c=new a(1);var d=new a(2);

c instanceof a為true而d instanceof b為false。

而用typeof c和typeof d的結果都是object

“判斷函數參數類型”需要根據你的需求來選擇用哪個。

您可能感興趣的文章:
  • javascript instanceof 與typeof使用說明
  • javascript instanceof,typeof的區別
  • 關於javascript中的typeof和instanceof介紹
  • javascript之typeof、instanceof操作符使用探討
  • JS中typeof與instanceof之間的區別總結
  • JavaScript中instanceof與typeof運算子的用法及區別詳細解析
  • 淺談javascript中的instanceof和typeof
  • 談談我對JavaScript中typeof和instanceof的深入理解

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.