js面向對象編程/原型鏈/繼承 —— javascript

来源:https://www.cnblogs.com/mu159/archive/2019/08/18/11367161.html
-Advertisement-
Play Games

目錄 js面向對象編程 js原型鏈 共用方法 原型繼承 class繼承 js面向對象編程 js原型鏈 共用方法 原型繼承 class繼承 js面向對象編程 js面向對象編程不同於 java 的類和對象 JavaScript 不區分類和實例的概念,而是通過原型(prototype)來實現面向對象編程。 ...


目錄

 

js面向對象編程

js面向對象編程不同於 java 的類和對象

JavaScript 不區分類和實例的概念,而是通過原型(prototype)來實現面向對象編程。

js聲明的構造函數,類似於普通函數的聲明,但又不同,

實例對象時,如果不寫new,就是一個普通函數,它返回 undefined。

但是,如果寫了new,它就變成了一個構造函數,它綁定的 this 指向新創建的對象,

並預設返回 this,也就是說,不需要在最後寫return this;。

 

js原型鏈

 代碼段一:

function Student(name){
    this.name = name;
    this.say = function(){
        console.log('my name:', this.name);
    }
}

let student1 = new Student('student1');
let student2 = new Student('student2');

console.log(student1.constructor === Student.prototype.constructor)        // true 

橙色箭頭表示原型鏈,其原型鏈為:

student1 --> Student.prototype --> Object.prototype --> null

當我們用 obj.xx 訪問一個對象的屬性時,JavaScript引擎先在當前對象上查找該屬性,

如果沒有找到,就到其原型對象上找,如果還沒有找到,就一直上溯到Object.prototype對象,

最後,如果還沒有找到,就只能返回undefined

 

共用方法

代碼段二:

function Student2(){
    this.say = function(){
        console.log('hi')
    }
}
console.log(new Student2().say === new Student2().say)

 結果:

false

實例化的對象方法,雖然方法名稱和代碼完全一樣,但是不同對象指向的不是同一個方法

需要創建一個共用的方法,

根據原型鏈圖,需要將這個共用方法聲明在 Student2 的原型對象上,

xxx.prototype.xxx = function(){}

function Student2(){
    this.say = function(){
        console.log('hi')
    }
}

Student2.prototype.publicSay = function(){
    console.log('public say');
}

console.log(new Student2().say === new Student2().say)
console.log(new Student2().publicSay === new Student2().publicSay)

 結果:

false
true

 

原型繼承

學過 java 的都知道,類的繼承通過 extends 會很容易實現,

但是 javascript原型繼承有點麻煩,不過 class繼承就很方便

function Father(name){
    this.say = function(){
        console.log(name)
    }
}
function Son(name){
    Father.call(this, name)
}

console.log(Son.prototype.__proto__)   // Object

這樣看似繼承了,但是其原型鏈的指向並沒有改變

其原型鏈圖為:

要實現原型繼承,看圖的話很容易,只需要將 Son 的原型對象的原型指向 Father 的原型對象

 

要實現原型繼承,這裡有三種方法,

 法一:

這個方法簡介明瞭,但是不推薦直接通過 __proto__ 直接改變原型

function Father(name){
    this.say = function(){
        console.log(name)
    }
}
function Son(name){
    Father.call(this, name)
}

Son.prototype.__proto__ = Father.prototype;
console.log(Son.prototype.__proto__)    // Father

 

法二:

通過實例化 Father 生成一個對象,

new Father() 的原型會預設指向 Father 的原型

通過修改 Sonprototype 屬性和 new Father()constructor 屬性,

來綁定 Sonnew Father() 之間的關係

function Father(name){
    this.say = function(){
        console.log(name)
    }
}
function Son(name){
    Father.call(this, name)
}

Son.prototype = new Father();
Son.prototype.constructor = Son;

console.log(Son.prototype.__proto__)    // Father

 

法三:

 類似法二,聲明一個中間對象來改變指向

Mid.prototype = Father.prototype;
Son.prototype = new Mid();
Son.prototype.constructor = Son;

第一步,將 Mid 的原型對象指向 Father 的原型對象,

第二步,將 Son 的屬性 prototype 指向 Mid

  此時代碼上的 new Mid(),實際上是 new Father()

第三步,將 Son.prototype.constructor 也就是 Mid.prototype.constructor 指向 Son

看起來有點亂,看數字步驟,方便理解

function Father(name){
    this.say = function(){
        console.log(name)
    }
}
function Son(name){
    Father.call(this, name)
}

Mid.prototype = Father.prototype;
Son.prototype = new Mid();
Son.prototype.constructor = Son;

console.log(Son.prototype.__proto__)  // Father

class繼承

ES6 提供了關鍵字 class,定義類變得更便捷

共用方法

class Father{
    // 構造方法
    constructor(name){
        this.name = name;
    }
    hello(){
        console.log("hello", this.name)
    }
}

console.log(new Father("f").hello === new Father("f").hello)
// true,共用方法

class繼承

class Father{
    // 構造方法
    constructor(name){
        this.name = name;
    }
    hello(){
        console.log("hello", this.name)
    }
}

class Son extends Father{
    constructor(name){
        super(name);
    }
}

console.log(Son.prototype.__proto__)    // Father

 


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

-Advertisement-
Play Games
更多相關文章
  • if(val.toString().length == 10){ val = val.toString().padEnd(13,"0")*1 //不夠十三位放後面補零,超過13位也可以 } toString() 方法可把一個邏輯值轉換為字元串,並返回結果。 ES2017 引入了字元串補全長度的功能。 ...
  • 0. Typescript Typescript對於前端來說可以說是越來越重要了,前端的很多項目都用Typescript進行了重構。這主要得益於Typescript有比較好的類型支持,在編碼的過程中可以很好地做一些類型推斷(主要是編輯器會有代碼提示,就很舒服)。再者Typescript的語法相較於j ...
  • 首先創建一個calculate.jsp 這是用Javascript代碼來驗證,代碼如下: <script type="text/javascript"> function check(){ if(form.n1.value==""&&form.value.n2==""){ window.alert( ...
  • Daily ,一個 寫的 android app 。 下拉刷新獲取:圖片、詩句、言語、音樂、樂評、雨聲、知乎日報、歷史上的今天。 可以說是一個 入門級 的 應用。 "項目地址:https://github.com/imguolao/Daily" 效果預覽 第三方 api 項目中的數據均來源於第三方 ...
  • 如果流圖片要載入失敗, 就會顯示找不到圖片的裂痕 代碼如下: 圖片目錄: 修改方法 : img標簽src為loading占點陣圖的地址,lay-src為正圖地址,圖片懶載入時會替換src<img src="loadingImg/loading.gif" lay-src="ddd.jpg"> 修改jsl ...
  • html代碼: js代碼: ...
  • layer彈出視窗在彈出時指定了area,彈出後,如果當前頁面(iframe)大小比彈出的視窗小,那麼就會出現無法操作彈出視窗的尷尬情況。如圖: 彈出視窗比當前頁面大,這時,唯有放大整個頁面才能看到完全的彈出視窗,才可以操作。 layui 為我們提供了 layer.style(); 方法來重新跳整窗 ...
  • 效果圖 ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...