成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Js基礎(chǔ)知識(二) - 原型鏈與繼承精彩的講解

mrcode / 401人閱讀

摘要:有了原型鏈,就有了繼承,繼承就是一個對象像繼承遺產(chǎn)一樣繼承從它的構(gòu)造函數(shù)中獲得一些屬性的訪問權(quán)。這里其實就是一個原型鏈與繼承的典型例子,開發(fā)中可能構(gòu)造函數(shù)復雜一點,屬性定義的多一些,但是原理都是一樣的。

作用域、原型鏈、繼承與閉包詳解
注意:本章講的是在es6之前的原型鏈與繼承。es6引入了類的概念,只是在寫法上有所不同,原理是一樣的。
幾個面試常問的幾個問題,你是否知道

instanceof的原理

如何準確判斷變量的類型

如何寫一個原型鏈繼承的例子

描述new一個對象的過程

也許有些同學知道這幾個問題的答案,就會覺得很小兒科,如果你還不知道這幾個問題的答案或者背后所涉及到的知識點,那就好好看完下文,想必對你會有幫助。先不說答案,下面先分析一下涉及到的知識點。
什么是構(gòu)造函數(shù)

JavaScript沒有類的概念,JavaScript是一種基于對象的語言,除了五中值類型(number boolean string null undefined)之外,其他的三種引用類型(object、Array、Function)本質(zhì)上都是對象,而構(gòu)造函數(shù)其實也是普通的函數(shù),只是可以使用構(gòu)造函數(shù)來實例化對象。

事實上,當任意一個普通函數(shù)用于創(chuàng)建一類對象時,它就被稱作構(gòu)造函數(shù)。像js的內(nèi)置函數(shù)Object、Array、Date等都是構(gòu)造函數(shù)。

在定義構(gòu)造函數(shù)有以下幾個特點:

以大寫字母開頭定義構(gòu)造函數(shù)

在函數(shù)內(nèi)部對新對象(this)的屬性進行設(shè)置

返回值必須是this,或者其它非對象類型的值

下面定義一個簡單的、標準的構(gòu)造函數(shù):

function Obj(){
    this.name = "name"
    return this // 默認有這一行 
}
var foo = new Obj() // 使用上面定義的構(gòu)造函數(shù)創(chuàng)建一個對象實例
原型特性

js原型有5個特點,記住這5條特點,相信你一定會弄明白長期困擾你的原型關(guān)系。

除了null所有引用類型(Object、Array、Function)都有對象特性,也就是都可以自由擴展屬性。

所有引用類型都有一個_proto_屬性(又稱為:隱式屬性),_proto_是一個普通的對象。所有的對象都會有一個constructor屬性,constructor始終指向創(chuàng)建當前對象的構(gòu)造函數(shù)

所有的函數(shù)都有一個prototype屬性(又稱為:顯式屬性),也是一個普通對象,這個prototype有一個constructor屬性指向該函數(shù)。

所有的引用類型的_proto_屬性指向它的構(gòu)造函數(shù)的prototype屬性(比如:obj._proto_指向Object.prototype,obj是定義的一個普通對象,Object是js的內(nèi)置函數(shù))

當從一個對象中獲得某個屬性時,如果這個對象沒有該屬性,就會去它的_proto_(也就是它的構(gòu)造函數(shù)的prototype)中去尋找

先來解釋一下這幾條:
第一條的自由擴展性可以通過一個簡單的例子來看

var obj = {}
obj.name = "name"
console.log(obj) // {name:"name"}

第二條和第三條是javascript就是這么規(guī)定的,沒什么好說的

第四條可以這么理解,當定義一個引用類型的變量var obj = {} 其實是var obj = new Object()的語法糖,這樣Object就是obj的構(gòu)造函數(shù),根據(jù)第4條規(guī)定,obj._proto_ === Object.prototype,如果不理解可以看看上一章我們講的js內(nèi)置函數(shù)和上面講的構(gòu)造函數(shù)

第五條應(yīng)該好理解,當從obj中獲取某個屬性時,如果obj中沒有定義該屬性,就會逐級去它的_proto_對象中去尋找,而它的_proto_指向Object的prototype,也就是從Object的prototype對象中去尋找。

原型鏈與繼承
如果上面明白了原型,那么原型鏈就會很好理解

根據(jù)原型定義的第4條和第5條,很容易發(fā)現(xiàn)通過對象的_proto_和函數(shù)的prototype把我們變量和構(gòu)造函數(shù)(自定義的構(gòu)造函數(shù)以及內(nèi)置構(gòu)造函數(shù))像鏈子一樣鏈接起來,所以又叫他原型鏈。

有了原型鏈,就有了繼承,繼承就是一個對象像繼承遺產(chǎn)一樣繼承從它的構(gòu)造函數(shù)中獲得一些屬性的訪問權(quán)。從下面一個小例子理解:

function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};
// 原型繼承
function Cat(){ 
}
Cat.prototype = new Animal();
Cat.prototype.name = "cat";

上面例子中在Foo構(gòu)造函數(shù)的prototype中自定義一個somefn函數(shù)。然后通過new Foo()創(chuàng)建一個對象實例并賦值給bar變量,此時bar就等于{name:"bar"}。然后bar.somefn就去bar對象中尋找somefn這個屬性,發(fā)現(xiàn)找不到,然后就去它的_proto_(其實就是Foo的prototype)中尋找,發(fā)現(xiàn)somefn就在Foo的prototype中定義了,就可以愉快的調(diào)用并執(zhí)行somefn了。

這里其實就是一個原型鏈與繼承的典型例子,開發(fā)中可能構(gòu)造函數(shù)復雜一點,屬性定義的多一些,但是原理都是一樣的。

留一個問題,根據(jù)上面例子,如果執(zhí)行bar.stString(),應(yīng)該去哪里找toString這個方法? (提示:prototype也是普通對象,也有自己的_proto_)
幾種繼承方式

這幾種都是es5中的繼承,es6中提供了class類,繼承起來更方便。

原型繼承

上述例子就是一個原型繼承:

function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};
// 原型繼承
function Cat(){ 
}
Cat.prototype = new Animal();
Cat.prototype.name = "cat";

var cat = new Cat()
console.log(cat instanceof Animal); //true 
console.log(cat instanceof Cat); //true

優(yōu)點:

非常純粹的繼承關(guān)系,實例是子類的實例,也是父類的實例

簡單,易于實現(xiàn)

缺點

要想為子類新增屬性和方法,必須要在new Animal()這樣的語句之后執(zhí)行,不能放到構(gòu)造器中

無法實現(xiàn)多繼承

來自原型對象的引用屬性是所有實例共享的(嚴重缺點)

創(chuàng)建子類實例時,無法向父類構(gòu)造函數(shù)傳參(嚴重缺點)

構(gòu)造繼承
function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};
// 構(gòu)造繼承
function Cat(name){
  Animal.call(this);
  this.name = name || "Tom";
}

// Test Code
var cat = new Cat();
console.log(cat.name);
console.log(cat.sleep()); // Tom正在睡覺!
// console.log(cat.eat("fish")); // cat.eat is not a function
console.log(cat instanceof Animal); // false
console.log(cat instanceof Cat); // true

優(yōu)點

解決了1中,子類實例共享父類引用屬性的問題

創(chuàng)建子類實例時,可以向父類傳遞參數(shù)

可以實現(xiàn)多繼承

缺點

實例并不是父類的實例,只是子類的實例

只能繼承父類的實例屬性和方法,不能繼承原型屬性/方法

無法實現(xiàn)函數(shù)復用,每個子類都有父類實例函數(shù)的副本,影響性能

實例繼承
function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};

// 實例繼承
function Cat(name){
  var instance = new Animal();
  instance.name = name || "Tom";
  return instance;
}
var cat = new Cat(); // 或者可以直接var cat = Cat()
console.log(cat.name);
console.log(cat.sleep()); // Tom正在睡覺!
console.log(cat.eat("fish")); // Tom正在吃:fish
console.log(cat instanceof Animal); // true
console.log(cat instanceof Cat); // false

優(yōu)點

不限制調(diào)用方式,不管是new Cat()還是Cat(),返回的對象具有相同的效果

缺點

實例是父類的實例,不是子類的實例

不支持多繼承

組合繼承
function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};
// 組合繼承
function Cat(name){
  Animal.call(this);
  this.name = name || "Tom";
}
Cat.prototype = new Animal();
Cat.prototype.constructor = Cat;
var cat = new Cat();
console.log(cat.name);
console.log(cat.sleep()); // Tom正在睡覺!
console.log(cat.eat("fish")); // Tom正在吃:fish
console.log(cat instanceof Animal); // true
console.log(cat instanceof Cat); // true

優(yōu)點

彌補了方式2的缺陷,可以繼承實例屬性/方法,也可以繼承原型屬性/方法

既是子類的實例,也是父類的實例

不存在引用屬性共享問題

可傳參

函數(shù)可復用

缺點

調(diào)用了兩次父類構(gòu)造函數(shù),生成了兩份實例(子類實例將子類原型上的那份屏蔽了)

寄生繼承
var ob = {name:"小明",friends:["小花","小白"]};

function object(o){
  function F(){}//創(chuàng)建一個構(gòu)造函數(shù)F
  F.prototype = o;
  return new F();
}

//上面再ECMAScript5 有了一新的規(guī)范寫法,Object.create(ob) 效果是一樣的  

function createOb(o){
   var newob = object(o);//創(chuàng)建對象
   newob.sayname = function(){//增強對象
       console.log(this.name);
   }

   return newob;//指定對象
}

var ob1 = createOb(ob);
ob1.sayname()
寄生繼承原理尚不明白。
寄生組合繼承

寄生組合繼承有兩種方式:

第一種:利用創(chuàng)建沒有實例方法的函數(shù)

function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};

//寄生組合繼承
function Cat(name){
  Animal.call(this);
  this.name = name || "Tom";
}
(function(){
  // 創(chuàng)建一個沒有實例方法的類
  var Super = function(){};
  Super.prototype = Animal.prototype;
  //將實例作為子類的原型
  Cat.prototype = new Super();
  Cat.prototype.constructor = Cat;
})();

var cat = new Cat();
console.log(cat.name);
console.log(cat.sleep()); // Tom正在睡覺!
console.log(cat.eat("fish")); // Tom正在吃:fish
console.log(cat instanceof Animal); // true
console.log(cat instanceof Cat); //true

第二種:利用Object.create函數(shù)

// 寄生繼承核心方法
function inheritPrototype(Parent, Children){
    var prototype = Object.create(Parent.prototype);
    prototype.constructor = Children;
    Children.prototype = prototype;
}
// 父類
function Animal (name) {
  // 屬性
  this.name = name || "Animal";
  // 實例方法
  this.sleep = function(){
    console.log(this.name + "正在睡覺!");
  }
}
// 原型方法
Animal.prototype.eat = function(food) {
  console.log(this.name + "正在吃:" + food);
};

// 子類
function Cat(name){
  Animal.call(this);
  this.name = name || "Tom";
}
inheritPrototype(Animal, Cat)

var cat = new Cat();
console.log(cat.name);
console.log(cat.sleep()); // Tom正在睡覺!
console.log(cat.eat("fish")); // Tom正在吃:fish
console.log(cat instanceof Animal); // true
console.log(cat instanceof Cat); //true

Object.create其實與以下代碼等價

function object(o){
    function f(){}
    f.prototype = o;
    return new f();
}

優(yōu)點

最完美的繼承解決方案

缺點

實現(xiàn)復雜

解答一下最一開始提出的問題

看到這里應(yīng)該對原型鏈與繼承的原理有所了解了,再回頭看上面的問題,你也會發(fā)現(xiàn)這都是小兒科。
第一個問題:instanceof原理?

var arr = []
arr instanceof Array

instanceof原理就是利用了原型鏈,當執(zhí)行arr instanceof Array時,會從arr的_proto_一層一層往上找,看是否能不能找到Array的prototype。
我們知道var arr = [] 其實是var arr = new Array()的語法糖,所以arr的_proto_指向Array的prototype,結(jié)果返回true

第二個問題:如何準確判斷變量類型?
可以使用instanceof幫助我們判斷,而不是typeof

第三個問題:如何寫一個原型鏈繼承的例子?

function Foo () {
    this.name = "name"
    this.run = function () {
        console.log(this.name)
    }
}
function Bar () {}
Bar.prototype = new Foo() // 從構(gòu)造函數(shù)Foo中繼承
var baz = new Bar()
baz.run() // 打印出 "name"

第四個問題:描述new一個對象的過程

創(chuàng)建一個新的對象,

獲得構(gòu)造函數(shù)的prototype屬性,并把prototype賦值給新對象的_proto_,this指向這個新對象

執(zhí)行構(gòu)造函數(shù),返回構(gòu)造函數(shù)的內(nèi)容

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/107143.html

相關(guān)文章

  • Js基礎(chǔ)知識) - 原型鏈與繼承精彩講解

    摘要:有了原型鏈,就有了繼承,繼承就是一個對象像繼承遺產(chǎn)一樣繼承從它的構(gòu)造函數(shù)中獲得一些屬性的訪問權(quán)。這里其實就是一個原型鏈與繼承的典型例子,開發(fā)中可能構(gòu)造函數(shù)復雜一點,屬性定義的多一些,但是原理都是一樣的。 作用域、原型鏈、繼承與閉包詳解 注意:本章講的是在es6之前的原型鏈與繼承。es6引入了類的概念,只是在寫法上有所不同,原理是一樣的。 幾個面試常問的幾個問題,你是否知道 insta...

    lingdududu 評論0 收藏0
  • JavaScript系列--淺析原型鏈與繼承

    摘要:綜上所述有原型鏈繼承,構(gòu)造函數(shù)繼承經(jīng)典繼承,組合繼承,寄生繼承,寄生組合繼承五種方法,寄生組合式繼承,集寄生式繼承和組合繼承的優(yōu)點于一身是實現(xiàn)基于類型繼承的最有效方法。 一、前言 繼承是面向?qū)ο螅∣OP)語言中的一個最為人津津樂道的概念。許多面對對象(OOP)語言都支持兩種繼承方式::接口繼承 和 實現(xiàn)繼承 。 接口繼承只繼承方法簽名,而實現(xiàn)繼承則繼承實際的方法。由于js中方法沒有簽名...

    draveness 評論0 收藏0
  • JavaScript系列(四) - 收藏集 - 掘金

    摘要:函數(shù)式編程前端掘金引言面向?qū)ο缶幊桃恢币詠矶际侵械闹鲗Х妒?。函?shù)式編程是一種強調(diào)減少對程序外部狀態(tài)產(chǎn)生改變的方式。 JavaScript 函數(shù)式編程 - 前端 - 掘金引言 面向?qū)ο缶幊桃恢币詠矶际荍avaScript中的主導范式。JavaScript作為一門多范式編程語言,然而,近幾年,函數(shù)式編程越來越多得受到開發(fā)者的青睞。函數(shù)式編程是一種強調(diào)減少對程序外部狀態(tài)產(chǎn)生改變的方式。因此,...

    cfanr 評論0 收藏0
  • JavasScript重難點知識

    摘要:忍者級別的函數(shù)操作對于什么是匿名函數(shù),這里就不做過多介紹了。我們需要知道的是,對于而言,匿名函數(shù)是一個很重要且具有邏輯性的特性。通常,匿名函數(shù)的使用情況是創(chuàng)建一個供以后使用的函數(shù)。 JS 中的遞歸 遞歸, 遞歸基礎(chǔ), 斐波那契數(shù)列, 使用遞歸方式深拷貝, 自定義事件添加 這一次,徹底弄懂 JavaScript 執(zhí)行機制 本文的目的就是要保證你徹底弄懂javascript的執(zhí)行機制,如果...

    forsigner 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<