創建對象

来源:http://www.cnblogs.com/yaoyinglong/archive/2016/04/04/5353353.html
-Advertisement-
Play Games

作者:禪樓望月( http://www.cnblogs.com/yaoyinglong ) JavaScript里也可以像Java等面向對象的語言世界里創建自定義的類型,但是由於JavaScript中不支持使用class關鍵字來創建自定義的類型,因此我們只能另闢蹊徑……下麵我們一起來看在JavaSc ...


作者:禪樓望月( http://www.cnblogs.com/yaoyinglong

JavaScript里也可以像Java等面向對象的語言世界里創建自定義的類型,但是由於JavaScript中不支持使用class關鍵字來創建自定義的類型,因此我們只能另闢蹊徑……下麵我們一起來看在JavaScript中如何使用OO。

1工廠模式

[+] view code function createPersoin(name,age){     var o=new Object();     o.name=name;     o.age=age;     o.sayName=function(){return o.name;}     return o; } var p1=createPersoin('yyl',27); var p2=createPersoin('dyy',27); 

調用createPerson就可以創建一個對象,看起來我們已經創建了一個自定義的類型,該類型可以用來表示Person,但是問題是,怎麼讓解析引擎知道這是一個Person呢?很抱歉,這種工程模式解決不了這種問題,所有的實例都是Object類型。

2.構造函數模式

為瞭解決對象識別的問題,JavaScript中又出現了一種新的模式,即這裡的構造函數模式,構造函數其實就是一個普通的函數,除了函數名稱首字母一般要大寫外,其他的和普通的函數沒什麼區別。我們用構造函數模式改造是上面的工廠模式:

[+] view code function Persoin(name,age){     this.name=name;     this.age=age;     this.sayName=function(){return this.name;} } var p1=new Persoin('yyl',27); var p2=new Persoin('dyy',27); 

與工程模式相比,這次我們沒有在構造函數中顯示地創建一個Object,也沒有return語句,並且把所有的屬性與方法賦值給this。而在創建實例的時候,使用了我們可愛的new操作符,那我們來看下new操作符是怎麼創建出一個Person的實例的:

  1. 構建一個新的對象,這時這個對象可不單單是Object了,它還是Person,將該對象的[[Prototype]]設置為對應構造函數的原型對象;
  2. 將構造函數的作用域賦值給該對象;
  3. 執行構造函數中的代碼;
  4. 返回該對象。

這樣構造出來的對象便可以通過instanceof操作符來識別了:

[+] view code alert(p1 instanceof Persoin);//true 

剛剛也說了,構造函數其實和普通的函數一模一樣,只是函數名稱首字母被大寫了而已。如果在調用構造函數的時候沒用使用new操作符,豈不是將所有的屬性和方法添加到window上了,這樣可不好,為此我們重寫Person的構造函數:

[+] view code function Persoin(name,age){     if(!(this instanceof Persoin)){         return new Persoin(name,age);     }     this.name=name;     this.age=age;     this.sayName=function(){return this.name;} } 

image

當丟失了new操作符的時候,我們強制返回一個Person對象,這樣全局作用域就不會被污染了。

構造函數模式的缺點:

所有的屬性和方法都不會被共用,p1和p2都是Person的對象,但是他們中的sayName雖然完成的功能相同,但卻是兩個不同的Function實例,這是非常沒有必要的。況且sayName內部有tihs,就更沒有必要在執行代碼前將函數綁定到特定的對象上面。

3.原型模式

在創建一個函數的時候,JavaScript引擎就會根據一組規則為該函數創建一個prototype屬性,該屬性是一個指針,指向一個對象,這個對象的用途是包含可以由特定類型的所有實例共用的屬性和方法。現在我們用原型模式來改造構造函數模式:

[+] view code function Persoin(name,age){     if(!(this instanceof Persoin)){         return new Persoin(name,age);     }     this.name=name;     this.age=age; } Persoin.prototype.sayName= function () {     return this.name; }; 

image

由上圖可發現,這個原型對象其實就是一個特殊的Person對象,特殊到哪裡了呢?

  1. 它是由JavaScript引擎根據一定的規則幫我們創建的;
  2. 預設只包含constructor屬性(其他屬性,方法都是從Object繼承而來),而這個constructor指向包含該prototype的函數(這裡便是Person函數);
  3. 強制將該原型對象的[[Prototype]]指向Object函數的原型對象。

現在我們就把要在各個實例中共用的屬性和方法添加到該對象中。

image

它是怎麼做到在各個實例中共用的呢?由159行的代碼,我們可以看出,每個Person的實例都包含一個指向Person的原型的指針(__proto__),嗯,就是這樣做到共用的。

有沒有註意到剛剛所說的這個prototype的用途?它只是用來存儲包含它的構造函數的所有實例所共有的屬性和方法。那它就沒必要必須是一個特殊的Person對象了,並且每次為其添加屬性和方法總是要寫Person.prototype.……很麻煩,因此我們可以這樣寫:

[+] view code function Persoin(name,age){     if(!(this instanceof Persoin)){         return new Persoin(name,age);     }     this.name=name;     this.age=age;     //this.sayName=function(){return this.name;} } /*Persoin.prototype.sayName= function () {     return this.name; };*/ Persoin.prototype={     sayName: function () {         return this.name;     },     sayAge: function () {         return this.age;     } }; //查看原型對象 var p=Persoin.prototype; //創建實例 var p1=new Persoin('yyl',27); var p2=new Persoin('dyy',27); var flag=(p1.__proto__===p2.__proto__); 

image

現在,Person構造函數的原型已經不再是一個Person的特殊實例,而是一個Object。現在有一個問題,這個Object對象中沒有constructor屬性了。但是還是能訪問到這個屬性:

onsole.info(p.constructor)//function Object()

註意這可以不是該Person的原型對象中的constructor,它是Object的原型對象的constructor。

image

為什麼能拿到Object類型的原型對象的constructor屬性呢?這是因為在JavaScript獲取屬性和方法的值的過程其實就是一次搜索的過程,看一個更明顯的例子:

console.info(str=p1.toString());//[object Object]

我們並沒有在構造函數中定義toString函數,也沒有再Person的原型中定義該函數,那是怎麼獲取的呢?

image

首先在構造函數中找,接著在原型對象的中找,再在該原型對象所在函數的原型對象中找,以此類推,直到找到為止。

如果這個constructor的值很重要的話,添加上就是了,但是這會導致其[[Enumerable]]特性被設置為true。而原生的該特性的值為false。

最重要的一句話:原型對象是引用類型

4.動態原型模式

如果你是用過Java、C#或者其他的面向對象的語言,你可能覺得這種將構造函數和原型獨立起來的做法很彆扭,那麼動態原型模式就是解決這一問題的:

[+] view code function Persoin(name,age){     if(!(this instanceof Persoin)){         return new Persoin(name,age);     }     this.name=name;     this.age=age;     if(typeof this.sayName!=="function"){         Persoin.prototype.sayName= function () {             return this.name;         };         Persoin.prototype.sayAge= function () {             return this.age;         }     } } //查看原型對象 var p=Persoin.prototype; //創建實例 var p1=new Persoin('yyl',27); 

image

這種方法構造對象可以說是非常完美的。註意if語句可以是初始化之後應該存在的任何屬性或方法,但是請註意,不必用一大堆的if來檢查每個屬性和方法,只要檢查其中一個即可。它是用來說明在第一次調用構造函數的時候執行if裡面的代碼,以後就不要再執行這些代碼了。

特別註意:使用動態原型模式構造對象時,不能使用對象字面量來重寫原型。這是又是為什麼呢?

image

我們來看一下這串代碼的執行過程,首先JavaScript引擎會將Person函數放在最開始創建出來,這是它的原型對象便是我們所說的那個特殊的Person對象(正如在161行代碼獲取到的一樣),執行到163行代碼,JavaScript得到的指令時去新建一個Person實例,回想一下JavaScript創建對象的過程:1.創建一個新的對象;對了,問題的關鍵就在這裡了,JavaScript引擎創建的這個對象中已經包含了[[Prototype]]這個內部指針。它現在指向這個特殊的Person實例。2.將構造函數的作用域賦值給該對象;3.執行構造函數中的代碼,當執行了151-157的代碼時,Person.prototype的指向已將變了,現在它指向這個對象字面量了。而目前的p1的[[Prototype]]還是指向原來的原型對象。這就導致了我們從p1中訪問不到sayName方法。當執行164行代碼時,JavaScript引擎再去新建一個Person實例,這時Person.prototype已經在新建p1時被替換了,它順理成章的擁有了這個原型對象。當執行到151-157的代碼時由於原型對象中有sayName方法,這段代碼沒有被執行,因此,創建出來的p2才是符合我們願望的對象。


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

-Advertisement-
Play Games
更多相關文章
  • Java虛擬機位元組碼指令 瞭解了class文件,我覺得就很有必要去瞭解一下JVM中的位元組碼指令,那樣堆class文件以及JVM運行機制也後很大的幫助. Java虛擬機的指令由一個位元組長度的,代表著某種特定操作含義的數字(稱為操作碼,Opcode)以及跟隨其後的零至多個代表所需參數(稱為操作數,Opr ...
  • 一.String類概述 1.String的值是一個對象,也是一個常量不能被改變 2.String的equals方法是比較兩個字元串的內容 3.String s1=”abc” String s2=new String(“abc”) s1有一個對象,s2有兩個對象 二.String常見功能 獲取和判斷 ...
  • 屬性表集合 在前面魔數,次版本號,主板本號,常量池入口,常量池,訪問標誌,類索引,父類索引,介面索引集合,欄位表集合,方法表集合,那麼接下來就是屬性表集合了. 屬性表集合 在前面魔數,次版本號,主板本號,常量池入口,常量池,訪問標誌,類索引,父類索引,介面索引集合,欄位表集合,方法表集合,那麼接下來 ...
  • 構造函數基礎 構造函數是一種特殊的函數,主要用來在創建對象時初始化對象,為對象成員變數設置初始值,在 OC 中的構造函數是 initWithXXX,在 Swift 中由於支持函數重載,所有的構造函數都是 init 構造函數的作用 分配空間 alloc 設置初始值 init 必選屬性 自定義 Pers ...
  • 問題場景 最近項目在發佈的時候出現了一個莫名其妙的錯誤:在websphere8.5上面可以部署但是不能正常訪問,但是在tomcat和websphere7.0上面卻可以正常部署訪問。所報錯誤如下: 剛開始以為是版本相容性的問題,於是到網上各種找答案,還是不能解決。後來順著filter這條線索發現,應該 ...
  • dict是字典dictionary的縮寫,他存放的是鍵值對key/value,用花括弧表示,格式為d={'micheal':99,'jack':88} 當我們訪問的時候直接print(d['micheal'])#輸出相應的99; 可以給dict複製並且會覆蓋掉原來的value,但是如果key沒有對應 ...
  • 隨著互聯網信息技術日新月異的發展,一個海量數據爆炸的時代已經到來。如何有效地處理、分析這些海量的數據資源,成為各大技術廠商爭在激烈的競爭中脫穎而出的一個利器。可以說,如果不能很好的快速處理分析這些海量的數據資源,將很快被市場無情地所淘汰。當然,處理分析這些海量數據目前可以借鑒的方案有很多:首先,在分 ...
  • 回顧: 上一節中鄙人通過解析表達式樹生成Sql碎片,其中我也把解析表達式類代碼貼了出來,文章發佈之後我對ExpressionAnalyzer類做了些改動,下麵我還會將代碼貼出來,廢話不多說,直接進入今天的主題。 實體類設計: 首先,我覺得要想直接通過實體類生成Sql語句,那麼你可能要知道這個實體類對 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...