js继承

来源:互联网 发布:如果女朋友是病娇 知乎 编辑:程序博客网 时间:2024/06/18 15:24

继承

1.所有的函数都有prototype属性,指向原型对象,所有的对象都有一个constructor属性,指向构造函数。
从技术上来讲,js上并不存在继承。js本身不存在面向对象特性,是面向过程的语言。

实现继承的方式(1)原型链继承

var b.prototype = new a();

好处:
(1)比较简单;
(2)在父类的原型中,可以动态的增加属性和方法,而不会影响到a。
缺点:
(1)为子类添加属性和方法时,必须写在var b.prototype = new a();之后,否则,创建b时就是覆盖之前添加的属性和方法。
(2)使用原型链式的继承,不能实现多继承。(不是用的特别多)
(3)所有的属性都是共享的。(不是很严重)
(4)没有办法传递参数。(非常重要的缺点)

解决方式:借用构造函数模式(将父类中的属性重写一遍,在子类中重新运行一遍)

修改this的指向,a.call(this) 或者a.apply(this)。
实际开发代码: 在b中写入a.call(this) 或者a.apply(this)。

function SuperType(){    this.colors = ["red","blue","green"];}function SubType(){    //继承了SuperType    SuperType.call(this);}//SubType.prototype = new SuperType();var instance1 = new SubType();instance1.colors.push("orange");console.log(instance1.colors); //[ 'red', 'blue', 'green', 'orange' ]var instance2 = new SubType();console.log(instance2.colors);//[ 'red', 'blue', 'green' ]

传递参数

function SuperType(name){    this.name = name;}function SubType(){    //继承了SuperType    SuperType.call(this,"lulu");    this.age = 28;}//SubType.prototype = new SuperType();var instance = new SubType();console.log(instance.name);  //luluconsole.log(instance.age);   //28

优点

(1)可以实现多继承
(2)解决了共享问题
(3)可以传递参数

缺点

(1)通过new创建出来的实例只是子类的实例,不是父类的实例。
(2)只能继承构造函数内的属性和方法,不能继承原型中的属性和方法。
(3)所有的属性都是在构造函数中运行的,没法进行复用。

组合模式(伪经典继承)

1.在子类的构造函数中使用call或apply方法,让父类的构造函数重新运行。
2.让子类的原型指向父类的实例。

function SuperType(name){    this.name = name;    this.colors=["red","green","blue"];}SuperType.prototype.sayName = function(){    console.log(this.name);}function SubType(name,age){    //继承了SuperType    SuperType.call(this,name);    this.age = age;}//继承方法SubType.prototype = new SuperType();SubType.prototype.constructor = SubType;SubType.prototype.sayAge = function(){    console.log(this.age);};var instance1 = new SubType("lulu","28");instance1.colors.push("black");console.log(instance1.colors);  //[ 'red', 'green', 'blue', 'black' ]instance1.sayAge(); //28instance1.sayName(); //luluvar instance2 = new SubType("lpy","26");console.log(instance2.colors);//[ 'red', 'green', 'blue' ]instance2.sayAge();   //26instance2.sayName();  //lpy

以上三种方式,有一个共同的前提,就是所有的实例都是通过new创建出来的。
属性在实例和构造函数中各存在一份。

原型式继承

与原型链继承的区别是:原型链继承继承的是构造函数,原型式继承继承的是普通的函数。

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

从上面的例子可以看出,在object()函数内部,先创建了一个临时性的函数,然后将传入的对象作为这个构造函数的原型,最后返回了这个临时类型的一个新实例。

在ES5中,新增的创建对象方法Object.create(),这个方法接收两个参数:一个用作新对象原型的对象和一个为新对象定义额外属性的对象。在传入一个参数的情况下,Object.create() 与object()方法的行为相同。

//原型式继承var person = {    name:"露露",    friends:["李白","橘右京","阿珂"]};var anotherperson = Object.create(person);anotherperson.name = "花木兰";anotherperson.friends.push("杨戬");var yetAnotherperson = Object.create(person);yetAnotherperson.name="刘备";yetAnotherperson.friends.push("关羽");console.log(person.name+' '+person.friends);//露露 李白,橘右京,阿珂,杨戬,关羽console.log(anotherperson.name+' '+anotherperson.friends);//花木兰 李白,橘右京,阿珂,杨戬,关羽console.log(yetAnotherperson.name+' '+yetAnotherperson.friends);//刘备 李白,橘右京,阿珂,杨戬,关羽

Object.create()的第二个参数的每个属性都是通过自己的描述符定义的。

寄生式继承和原型式继承而思路一样,只不过看起来更像继承。

寄生组合式继承(经典继承)

针对组合模式的缺点(两次调用父类构造函数)我们使用寄生组合式继承(经典继承)。

所谓寄生组合式继承,即通过借用构造函数来继承属性,通过原型链的混成形式来继承方法。

function inheritPrototype(SubType,superType){    var p = Object(superType.prototype);    p.constructor = subType;    subType.prototype =p;   }

浅克隆:直接拷贝值(对于对象,则拷贝了引用)
深克隆:先判断数据类型,如果是原始类型,则直接拷贝,如果是对象,则赋值它的子对象的引用,里面所有的变量和子对象都是又额外拷贝了一份。

原创粉丝点击