javascript中的繼承

来源:http://www.cnblogs.com/yzg1/archive/2016/03/18/5292176.html
-Advertisement-
Play Games

我的博客: "http://bigdots.github.io" 、 "http://www.cnblogs.com/yzg1/" 繼承有什麼好處?很簡單,繼承了你爸的財產,自己就可以少奮鬥點嘛。開玩笑,言歸正傳,繼承使子類擁有超類的作用域、屬性與方法,可以節省程式設計的時間。ECMAScript實


我的博客:http://bigdots.github.iohttp://www.cnblogs.com/yzg1/

繼承有什麼好處?很簡單,繼承了你爸的財產,自己就可以少奮鬥點嘛。開玩笑,言歸正傳,繼承使子類擁有超類的作用域、屬性與方法,可以節省程式設計的時間。ECMAScript實現繼承主要方式是依靠原型鏈。

原型鏈方式——不建議使用

function SuperType(){
    this.colors = ["red", "blue", "green"];
}
function SubType(){
}
//利用原型鏈繼承
SubType.prototype = new SuperType();
var instance1 = new SubType();
console.log(instance1.colors) //["red", "blue", "green"]

上例中,通過SubType.prototype = new SuperType()使SubType的原型鏈上增加了SuperType()這一環。從而使SubType 通過原型鏈繼承了SuperType的color屬性。

存在的問題:

  1. 實例共用
    在上面的例子中加入以下代碼,發現SubType()的實例1對color屬性做出改變後,實例2獲取到的color屬性是改變後的值。這是因為:

    每個構造函數都有一個原型對象,原型對象都包含一個指向構造函數的指針,而實例都包含一個指向原型對象的內部指針。

    instance1.colors.push("black");
    console.log(instance1.colors); //["red", "blue", "green", "black", "black"]
    var instance2 = new SubType();
    console.log(instance2.colors);//["red", "blue", "green", "black", "black"]
  • 首先,instance1和instance2都是SubType()的實例,這倆個實例都會包含一個指向原型對象的內部指針
  • SubType()的原型指向SuperType()的一個實例,而且這個實例指向的是SuperType()的原型對象
  • SuperType()的原型對象又指向SuperType()。最終,instance1.colors和instance2.colors的指向都是SuperType()的colors。
  1. 沒有辦法在不影響所有對象實例的情況下,給超類型的構造函數傳遞參數,原理同上。

構造函數方式——不建議使用

function SuperType(){
    this.colors = ["red", "blue", "green"];
}
function SubType(){
    //繼承了SuperType
    SuperType.call(this);
}
var instance1 = new SubType();
console.log(instance1.color)//["red", "blue", "green"]

基本思想即在子類型構造函數的內部調用超類型構造函數。

call和apply
全局函數apply和call可以用來改變函數中this的指向。

存在的問題:

  1. 子類型無法繼承超類原型鏈,導致所有類型都只能使用構造函數模式;
  2. 方法都在構造函數中定義,函數無法復用。

組合繼承——最常用的繼承模式

function SuperType(name){
    this.name = name;
    this.colors = ["red", "blue", "green"];
}
SuperType.prototype.sayName = function(){
    console.log(this.name);
};
function SubType(name, age){
    //繼承屬性
    SuperType.call(this, name);   //第二次調用SuperType()
    this.age = age;
}
//繼承方法
SubType.prototype = new SuperType(); //第一次調用SuperType()
SubType.prototype.constructor = SubType;
SubType.prototype.sayAge = function(){
    console.log(this.age);
};
var instance1 = new SubType("Nicholas", 29);
instance1.colors.push("black");
console.log(instance1.colors); //"red,blue,green,black"
instance1.sayName(); //"Nicholas";
instance1.sayAge(); //29
var instance2 = new SubType("Greg", 27);
console.log(instance2.colors); //"red,blue,green"
instance2.sayName(); //"Greg";
instance2.sayAge();

基本思想:

  • 通過原型鏈來繼承超類的sayName()方法;
  • 通過構造函數的方式來使SuperType()里的屬性私有化;

為什麼可以實現屬性私有化?

  • 第一次調用超類,SubType.prototype 會得到兩個屬性:name 和colors;它們都是SuperType 的實例屬性,位於SubType 的原型中;
  • 第二次調用超類,在新對象上創建了實例屬性name 和colors。於是,這兩個屬性就屏蔽了原型中的兩個同名屬性;

存在的問題:
無論什麼情況下,都會調用兩次超類型構造函數::一次是在創建子類型原型的時候,另一次是在子類型構造函數內部

原型式繼承

function object(o){
    function F(){}
    F.prototype = o;
    return new F();
}
var person = {
    name: "Nicholas",
    friends: ["Shelby", "Court", "Van"]
};
var anotherPerson = Object.create(person);
anotherPerson.name = "Greg";
anotherPerson.friends.push("Rob");
var yetAnotherPerson = Object.create(person);
yetAnotherPerson.name = "Linda";
yetAnotherPerson.friends.push("Barbie");
console.log(person.friends); //["Shelby", "Court", "Van", "Rob", "Barbie"]

ECMAScript 5 通過新增Object.create()方法規範化了原型式繼承。這個方法接收兩個參數:一個用作新對象原型的對象和(可選的)一個為新對象定義額外屬性的對象

  • 將person對象作為基礎對象
  • 把person對象傳入到object()函數中,然後該函數就會返回一個新對象F(),這個新對象將person 作為原型
  • 適用於讓一個對象與另一個對象保持類似的情況

存在同原型鏈方式一樣的問題

寄生式基礎

function object(o){
    function F(){}
    F.prototype = o;
    return new F();
}
function createAnother(original){
    var clone = object(original); //通過調用函數創建一個新對象
    clone.sayHi = function(){ //以某種方式來增強這個對象
    console.log("hi");
};
return clone; //返回這個對象
}
var person = {
    name: "Nicholas",
    friends: ["Shelby", "Court", "Van"]
};
var anotherPerson = createAnother(person);
anotherPerson.sayHi(); //"hi"
  • 將person對象作為基礎對象
  • 將基礎對象傳遞給object()函數,將返回的新對象賦值給clone;(示範繼承模式時使用的object()函數不是必需的;任何能夠返回新對象的函數都適用於此模式。)
  • 為clone 對象添加一個新方法sayHi(),最後返回clone 對象

寄生組合式繼承——被認為是引用類型最理想的繼承範式

function object(o){
    function F(){}
    F.prototype = o;
    return new F();
}
function inheritPrototype(subType, superType){
    var prototype = object(superType.prototype); //創建對象
    prototype.constructor = subType; //增強對象
    subType.prototype = prototype; //指定對象
}

function SuperType(name){
    this.name = name;
    this.colors = ["red", "blue", "green"];
}
SuperType.prototype.sayName = function(){
    console.log(this.name);
};
function SubType(name, age){
    SuperType.call(this, name); //調用一次SuperType()
    this.age = age;
}
inheritPrototype(SubType, SuperType);
SubType.prototype.sayAge = function(){
    console.log(this.age);
};

var instance1 = new SubType("Nicholas", 29);
instance1.sayName();
console.log(instance1.colors);
  • 將超類的原型對位基礎對象,並且傳遞給object()函數,返回新對象賦值給prototype;
  • 將子類的原型指向prototype,即超類的原型,這裡繼承了超類的sayName方法
  • 子類中利用構造方法使子類上創建了name和colors屬性

參考書籍:《javascript高級程式設計》

如果覺得本文不錯的話,幫忙點擊下麵的推薦哦,謝謝!


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

-Advertisement-
Play Games
更多相關文章
  • html css
  • 自誕生之初截止目前(2016年初),React可以說是前端界最流行的話題,如果你還不知道React是何物,你就該需要充充電了。 d3是由紐約時報工程師開源的一個繪製基於svg的數據可視化工具,是近幾年最流行的visualization工具庫之一。d3提供豐富的svg繪製API、動畫甚至佈局等功能,目
  • 最近公司網站在谷歌,火狐上測試都沒有問題,但是在ietest,ie6上出現相容問題 ,由於ietest好幾次打開ie6都報錯(嘗試卸載重新安裝幾次無果),下載virtualbox安裝自帶ie6的xp系統,進行ie6測試,打開各大公司網站總是關閉,懷疑可能是xp的iso鏡像問題,於是又重新下載裝了第二
  • 斷斷續續的把慕課的JavaScript基礎和進階看完了,期間不怎麼應用有的都忘記了,接下來多開始寫些效果,進行實際應用。 製作一個表格,顯示班級的學生信息。 1. 滑鼠移到不同行上時背景色改為色值為 #f2f2f2,移開滑鼠時則恢復為原背景色 #fff 2. 點擊添加按鈕,能動態在最後添加一行 3.
  • 這是一道面試題, 請先思考,在看講解 :) 講解如下: 1. main() , 列印的結果為: undefined , 1 a. > 來看第一個列印的值為什麼是 undefined。 在js中,方法和變數的聲明都是會提前的。也就是說不管你在何處聲明的方法或者變數,在js解析時,都會將其提前,具體看代
  • 外部裝備有兩種方法
  • (我會慢慢總結 這些以後 直接參考 不必百度谷歌 否則你一旦離開了百度谷歌 你啥都不是!) 知識擴展 Number()函數可以替代IsNaN()函數做是否非數字驗證,但是Number()返回NaN,IsNaN()返回boolean值。 有時候做字元串非空驗證可以用 “” 也可判斷Length是否為0
  • hover3d.js是一款效果超酷的滑鼠滑過圖片3D卡片效果jQuery插件。通過該插件可以製作出滑鼠滑過圖片時,圖片3D傾斜,旋轉的效果,非常有層次感。 使用該jQuery插件需要在頁面中引入jquery和jquery.hover3d.min.js文件 該滑鼠滑過圖片效果的HTML結構如下:包裹元
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...