#JavaScript對象與繼承

来源:http://www.cnblogs.com/seektruth/archive/2016/07/09/5656690.html
-Advertisement-
Play Games

JavaScript對象與繼承 JavaScript是我在C語言之後接觸的第二門編程語言,大一暑假的時候在圖書館找了一本中國人寫的 "JavaScript" 程式設計來看。那個時候在編程方面幾乎還是小白,再加上那本書根本沒有提JavaScript的編程機制,又有一些誤導性的話,一直以來對JavaSc ...


JavaScript對象與繼承

JavaScript是我在C語言之後接觸的第二門編程語言,大一暑假的時候在圖書館找了一本中國人寫的JavaScript程式設計來看。那個時候在編程方面幾乎還是小白,再加上那本書根本沒有提JavaScript的編程機制,又有一些誤導性的話,一直以來對JavaScript有很深的誤解,認為JavaScript只是一門在瀏覽器上運行的面向對象語言,值此文來寫下JavaScript當中很具有迷惑性和容易誤解的地方。當然限於作者水平有限,也沒有什麼開發經驗,所以難免有疏漏之處,還望批評指正。

JavaScript的對象

對象是什麼

JavaScript代碼當中隨處可見new關鍵字,很容易讓人產生誤解,認為JavaScript是Java一樣是基於類繼承的語言。但是事實並非如此,JavaScript當中並沒有類,那JavaScript的對象不是類那又是什麼呢?某種意義上說,JavaScript的對象就是Python當中的字典(哈希表),其實也就是類似這樣的鍵值對:

me={
    "fisrtName" : "seek",
    "lastName" : "truth" ,
    "getName" : function(){
        return this.firstName+this.lastName; //this相當於指向這個對象的指針
    }
}

這是一個比較有誤解性的地方,初次看到時候覺得有點無法理解,但仔細用一用還是覺得合理,我們既可以像Python一樣用[]運算符來獲取元素,也可以用.操作符來獲取元素:

me.firstName // => seek
me["lastName"] //=> truth
me.getName() // => seektruth

new運算符

既然JavaScript當中是沒有類的,那麼new運算符又是在乾什麼呢?這是JavaScript設計的最讓人誤解的地方之一。JavaScript是一門函數式編程語言,JavaScript當中函數是一等公民,JavaScript當中函數也是對象,函數對象在被創建的時候會被添加調用屬性,比較坑的是JavaScript函數有兩種調用方式,一種是加了new關鍵字的調用,一種是沒有new關鍵字的調用,前者會返回一個對象,後者會返回return語句當中的內容。考慮下麵的一段函數:

function Obj(name){
    this.name=name;
    return name;
}

如果我們用new運算符來調用:

obj = new Obj("seektruth") //obj會是一個對象:{"name": "seektruth"}

如果我們直接調用:

obj = Obj("seektruth") //obj會是一個字元串:"seektruth"

確實設計的挺坑的,我們在調用的時候需要分清楚是否需要使用new,一般來說需要用new關鍵字來調用的函數會採用大寫開頭。

還有更坑的是如果返回的返回值是一個對象:

function Obj(name){
    this.name=name;
    return {};
}

這樣無論我們是否用new運算符來調用都會返回return語句里的值:

new Obj("seektruth") //=> {}
Obj("seektruth") //=> {}

設計的是什麼鬼......

對象繼承

原型

前面已經說到過JavaScript當中是沒有類的,那JavaScript又是怎麼來實現繼承的呢?答案是通過原型鏈。在JavaScript當中,每個對象都會有一個原型,在創建對象的時候,如果不加說明的話,對象繼承的原型是Object.prototype,函數對象會繼承Function.prototype(Function.prototype繼承Object.prototype):

Object.prototype // => {}
Function.prototype // => [Function]

我們可以通過對象的__proto__熟悉來查看對象的原型:

a={}
a.__proto__  // => {}

JavaScript通過指定對象的原型來實現繼承,指定對象的原型主要有三種方式,一是在構造函數當中指明原型,二是直接修改對象的__proto__屬性,三是利用Object.create函數,下麵我們依次來看一看

在構造函數當中指定原型

我們可以在構造函數當中指定對象的原型:

me={
    "firstName" : "seek",
    "lastName" : "truth" ,
    "getName" : function(){
        return this.firstName+this.lastName; //this相當於指向這個對象的指針
    }
}

function Obj(name){
    this.firstName = name;
    this.__proto__ = me; //指定原型為me對象
}

指定了原型之後,我們新建了對象之後就可以訪問原型的屬性:

obj = new Obj("foo"); // => { firstName: 'foo' }
obj.firstName // => foo
obj.lastName // => truth
obj.getName() // => "footruth"

當訪問一個對象的時候,首先會嘗試在改對象當中尋找該屬性,如果沒有就回到原型當中尋找,直到Object.prototype。如果我們在新的對象當中重寫了原型當中的屬性(方法),那麼實際使用的時候我們新寫的屬性(方法)會覆蓋掉原型當中的定義,這有點像基於類的語言的函數重載。

註意如果原型me對象的lastname屬性有改變,因為obj對象是在原型當中尋找屬性,那麼這個obj對象的lastname屬性也會改變:

me.lastName = "me"
obj.lastName // => "me"
obj.getName() // =>  "foome"

直接改變對象的原型

我們也可以直接指定(改變)對象的原型:

obj2 = {}
obj2.__proto__ = me
obj2.firstName // => seek
obj2.lastName // => "me"
obj2.getName() // =>  "seekme"

使用Object.create函數

儘管說前兩種方法可以解決問題,但是這兩種寫法並不優雅,因為JavaScript並不是基於類的語言,第一寫法很容易給人以誤解,JavaScript語言精粹的作者Crockford認為new就不應該出現在JavaScript語言當中,而推薦使用Object.create函數來基於原型來創建對象。Object.create函數的用法很簡單:

obj3 = Object.create(me) // 以me為原型創建新的對象
obj3.firstName // => seek
obj3.lastName // => "me"
obj3.getName() // =>  "seekme"

obj3 = Object.create(me) 與obj2 = {};obj2.proto = me是等價的,但是前一種寫法更優雅也更易於理解。

總結

JavaScript作為一門基於原型的,函數式的編程語言在設計上有很多優雅與強大之處,但同時又有很多糟粕和坑,正式如此,JavaScript也是被誤解最多語言。學習了JavaScript的對象繼承機制,感覺自己的水平還是大有長進的。


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

-Advertisement-
Play Games
更多相關文章
  • 效果:(換膚出來一個div,選擇你想要的圖片,作為網頁背景,保存) 要點:cookie保存狀態 html代碼: css代碼: js代碼: 效果圖: ...
  • 前面的話 前端工程師最基本的工作是切圖。photoshop用的6不6,對於工作效率有很大的影響。小火柴將前端工程師需要掌握的photoshop的知識和技能進行了梳理和歸納,總結成以下目錄 目錄 前端工程師技能之photoshop巧用系列第一篇——準備篇 前端工程師技能之photoshop巧用系列第二 ...
  • × 目錄 [1]初始設置 [2]自動切圖 前面的話 隨著photoshop版本的不斷升級,軟體本身增加了很多新的功能,也為切圖工作增加了很多的便利。photoshop最新的版本新增了自動切圖功能,本文將詳細介紹photoshop的這個新功能 初始設置 當然首先還是要進行一些首選項設置 【1】在編輯 ...
  • 源碼: <input type="checkbox" id="cleckAll" />全選 <div class="list"> <input type="checkbox" />覆選一 <input type="checkbox" />覆選二 <input type="checkbox" />覆選 ...
  • Javascript小學生都知道了javascript中的函數調用時會 隱性的接收兩個附加的參數:this和arguments。參數this在javascript編程中占據中非常重要的地位,它的值取決於調用的模式。總的來說Javascript中函數一共有4中調用模式:方法調用模式、普通函數調用模式、 ...
  • 概述 上一篇我們介紹瞭如何將$.ajax和Vue.js結合在一起使用,並實現了一個簡單的跨域CURD示例。Vue.js是數據驅動的,這使得我們並不需要直接操作DOM,如果我們不需要使用jQuery的DOM選擇器,就沒有必要引入jQuery。vue-resource是Vue.js的一款插件,它可以通過... ...
  • 經過今天的討論, 我們組初步給出了設計方案。 安裝了一晚上的 mockupBuilder 終於繪製了幾份界面的原型圖, 這裡需要吐槽下 mockupBuilder, 這個軟體很好用, 不過網頁版和單機版提供的功能不太一樣, 感覺單機版更全一些, 但是缺點是無法保存。。。 首頁:(初步設想是類似網易雲 ...
  • 一、表單 <form id="" name="" method="post/get" action="負責處理的服務端"> id不可重覆;name可重覆;get提交有長度限制,並且編碼後的內容在地址欄可見,post提交無長度限制,且編碼後內容不可見。 </form> 1、文本輸入 文本框<input ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...