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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...