(四)继承

1.原型链

javascript中没有类的概念,需要利用原型链来模拟。
我们知道,构造函数、原型对象、实例之间有如下关系:

  1. 构造函数会自动获得一个prototype属性,指向其原型对象
  2. 原型对象在函数创建时自动获得,其中有一个constructor属性,指向其构造函数
  3. 实例中有一个__proto__指针,指向该类型的原型对象

如下图所示:

Paste_Image.png

原型链实现继承,就是将一个类型的原型(子类型)指向另一个类型(父类型)的一个实例。
我们知道,在寻找标识符(也就是变量或者函数)时,会先寻找实例属性,找到了即返回,如果没有找到,即通过__proto__指针去查找原型属性。
当用父类型的实例作为子类型的原型对象后,就可以找到父类型中的全部属性和方法,也就实现了继承,同时也切断了子类型构造函数与本来的原型对象的联系。如下图所示:

原型链继承过程
    //父类型构造函数
    function SuperType() {
        this.supername = "super"
    }
    //父类型原型对象
    SuperType.prototype.getSuperName = function () {
        console.log(this.supername)
    };
    //子类型构造函数
    function SubType() {
        this.subname = "sub";
    }
    //改变子类型原型属性的指向,图中1标记所示。
    //同时切断了子构造函数与它在创建时自动获得的原型对象的联系,如图中红叉所示。
    SubType.prototype = new SuperType();
    //添加子类型的方法
    SubType.prototype.getSubName = function () {
       console.log(this.supername)
    };
    var subInstance = new SubType();
    subInstance.getSuperName();//输出super,获得了父类型的方法

有一点需要注意,所有的实例都是Object类型的实例,这是因为函数创建时直接获得的原型对象的__proto__指针都是指向的Object.prototype

加入Object.prototype的原型链

而此时,子类型的实例又可以作为另一个类型(孙子)的原型对象,孙子类型的实例又可以作为曾孙类型的原型对象,由此形成一个链条。

纯原型链继承的问题
一个类型(父)的实例作为另一个类型(子)的原型对象,那么父类型的实例属性就变成了子类型实例的原型属性,如果这属性恰好是一个引用类型,那么这个属性所对应的引用就会被所有子类实例所共享。一个子类型实例对这个属性的任何操作都会影响其他所有的子类型实例。

2.借用构造函数实现继承

暗中观察一个构造函数:

  function Person(name,age,job) {
       this.name = name;
       this.age = age;
       this.job = job;
       this.sayName = function () {
           alert(this.name);
       };
   }
  var person1 = new Person();

可以看到,构造函数中的属性都定义在this上。回忆一下使用new操作符创建一个实例时的步骤:新建一个对象;然后让this指向这个对象,执行构造函数;构造函数执行完后,这个对象上就有创建好的各种属性;最后返回这个对象。

同样的道理,我们可以在子类型构造函数中使用call()apply()方法来改变构造方法的作用域,使其this指向子类构造函数的作用域,达到在子类型中创建父类型中定义的各种属性的目的。

  //示例代码来自《JavaScript高级程序设计》
  function SuperType(name) {
       this.name = name;
   }
   function SubType() {
       SuperType.call(this,"gordenz");
       this.age = "29";
   }
   var instance = new SubType();
   console.log(instance.name); //gordenz,获得了定义在父类中的属性
   console.log(instance.age); //29

call()apply()的区别:第一个参数都是函数执行的作用域 ;apply()的第二个参数接收一个参数数组,而call()方法则需要将接收的参数一一列在后面(即除了第一个参数外,可能有很多个参数)

全部使用构造函数实现继承的问题
1.无法实现同一函数的复用,每个实例都会创建一个副本;
2.子类型无法继承到定义在父类型的原型对象中的属性和方法。

3.组合继承

组合继承可以类比创建对象时的构造函数模式与原型模式。
即,使用原型链继承原型属性和方法使用构造函数继承实例属性

  //示例代码来自《JavaScript高级程序设计》
  function SuperType(name) {
       this.name = name;
       this.colors = ["red","blue","green"]
   }
   SuperType.prototype.sayName = function () {
       alert(this.name)
   };
   function SubType(name,age) {

       //继承属性
       SuperType.call(this,name);

       this.age = age;
   }

   //继承方法
   SubType.prototype = new SuperType();

   //为作为原型对象的父类型实例添加constructor属性
   SubType.prototype.constructor = SubType;
  
   //添加属于子类的原型方法
   SubType.prototype.sayAge = function () {
       alert(this.age);
   };

   var instance1 = new SubType("gordenz",24);
   instance1.colors.push('black');
   console.log(instance1.colors); //"red,blue,green,black"
   instance1.sayName(); //gorden
   instance1.sayAge(); //24

   var instance2 = new SubType("huan",23);
   console.log(instance2.colors); //"red,blue,green"
   instance2.sayName(); //huan
   instance2.sayAge(); //23 

   //父类的原型方法都得到了共享,而引用类型的属性也不会在各实例间相互影响

这样就完成了一个较好的继承,两个实例都拥有自己的属性,同时又公用了方法。
相对之前的原型链继承,此处多了一个步骤,就是将父类型的实例作为子类型的原型对象后,为其手工添加了一个constructor属性,使其指向了我们逻辑上的构造函数。如图:

Paste_Image.png

如图中2所示,我们手工为子类型的原型对象(它也是一个父类型的实例)指定了constructor属性,让他正确的指向子类型的构造函数。
如果我们没有手动指定这个constructor属性,那么此时他其实是指向的父类的构造函数。为什么呢?因为当前子类原型对象是一个父类型的实例,这个constructor属性继承自父类原型对象,所以他的指向是父类构造函数,这与我们的逻辑不符。
同时图中也展示了原型链的终点:null。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容