寄生组合式继承的优势详解

在之前javascript面向对象系列的文章里面,我们已经探讨了组合继承和寄生继承,回顾下组合继承:

function Person( uName ){
      this.skills = [ 'php', 'javascript' ];
      this.userName = uName;
    }
    Person.prototype.showUserName = function(){
      return this.userName;
    }
    function Teacher ( uName ){
      Person.call( this, uName );
    }
    Teacher.prototype = new Person();

    var oT1 = new Teacher( 'ghostwu' );
    oT1.skills.push( 'linux' );
    var oT2 = new Teacher( 'ghostwu' );
    console.log( oT2.skills ); //php,javascript
    console.log( oT2.showUserName() ); //ghostwu

组合继承有个缺点,父类的构造函数会被调用两次.

第11行,设置子类原型对象(prototype),调用了第一次

第9行,实例化对象的时候,又调用一次

构造函数的目的是为了复制属性,第9行肯定是不能少的,第11行的目的是为了获取到父类原型对象(prototype)上的方法,基于这个目的,有没有别的方法

可以做到 在不需要实例化父类构造函数的情况下,也能得到父类原型对象上的方法呢? 当然可以,我们可以采用寄生式继承来得到父类原型对象上的方法

function Person( uName ){
      this.skills = [ 'php', 'javascript' ];
      this.userName = uName;
    }
    Person.prototype.showUserName = function(){
      return this.userName;
    }
    function Teacher ( uName ){
      Person.call( this, uName );
    }

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

    function inheritPrototype( subObj, superObj ){
      var proObj = object( superObj.prototype ); //复制父类superObj的原型对象
      proObj.constructor = subObj; //constructor指向子类构造函数,避免重写而失去constructor
      //为了解决组合式继承中构造函数被执行两次的问题,
//我们将指向父类实例改为指向父类原型, 减去一次构造函数的执行
      subObj.prototype = proObj; //再把这个对象给子类的原型对象
    }

    inheritPrototype( Teacher, Person );

    var oT1 = new Teacher( 'ghostwu' );
    oT1.skills.push( 'linux' );
    var oT2 = new Teacher( 'ghostwu' );
    console.log( oT2.skills ); //php,javascript
    console.log( oT2.showUserName() ); //ghostwu

其实,说白了寄生组合式继承就是一个借用构造函数 + 相当于浅拷贝父类的原型对象


其他方案:

寄生组合式继承, 即通过借用构造函数来继承属性, 在原型上添加共用的方法, 通过寄生式实现继承

			//寄生组合式继承, 即通过借用构造函数来继承属性, 在原型上添加共用的方法, 通过寄生式实现继承.
			// 上边的例子高效的体现了只调用了一次 SuperType 构造函数,并且因此也避免了在 SubType.prototype 上面创建不必要的 多余的属性.与此同时,原型链还能保持不变
			// function inheritPrototype(subType, superType) {
			// 	var prototype = Object.create(superType.prototype); // 创建对象,创建父类原型的一个副本
			// 	prototype.constructor = subType; // 增强对象,弥补因重写原型而失去的默认的constructor 属性
			// 	subType.prototype = prototype; // 指定对象,将新创建的对象赋值给子类的原型
			// }
 
 
			// 父类初始化实例属性和原型的属性和方法
			function SuperType(name) {
				this.name = name;
				this.colors = ["red", "blue", "green"];
				this.sheet = function(sheet) {
					console.log(this.name + '正在sheet!' + sheet);
				}
			}
			SuperType.prototype.sayName = function() {
				console.log(this.name);
			};
 
			// 借用构造函数继承构造函数的实例的属性(解决引用类型共享的问题)
			function SubType(name, age) {
				SuperType.call(this, name);
				this.age = age;
			}
 
			(function(SubType, SuperType) {
				var prototype = Object.create(SuperType.prototype) //创建父类原型的一个副本
				prototype.constructor = prototype // 增强对象,弥补因重写原型而失去的默认的constructor 属性
				SubType.prototype = prototype //指定对象,讲新创建的对象赋值给子类的原型
			}(SubType, SuperType))
			// 将子类型的原型重写替换成父类的原型
			
			// inheritPrototype(SubType, SuperType);
 
			// 对子类添加自己的方法
			SubType.prototype.sayAge = function() {
				console.log(this.age);
			};
 
			var instance1 = new SubType("heyushuo", 12);
			var instance2 = new SubType("kebi");
			console.log('instance1', instance1)
			console.log('instance2', instance2)
			instance1.sayName(); //heyushuo
			instance1.sayAge(); //12
			instance2.sayName(); //kebi
 
			instance1.colors.push("yellow"); // ["red", "blue", "green", "yellow"]
			instance1.colors.push("black"); // ["red", "blue", "green", "black"]



以上这篇[js高手之路]寄生组合式继承的优势详解就是小编分享给大家的全部内容了

返回顶部
跳到底部

Copyright 2011-2024 南京追名网络科技有限公司 苏ICP备2023031119号-6 乌徒帮 All Rights Reserved Powered by Z-BlogPHP Theme By open开发

请先 登录 再评论,若不是会员请先 注册