JavaScript实现继承的方式优化
来源:互联网 发布:生活垃圾的数据 编辑:程序博客网 时间:2024/06/05 19:24
组合继承
将原型链和借用构造函数的技术组合在一块,从而发挥两者之长的一种继承模式。
var Parent = function(name){ this.name = name || 'parent';};Parent.prototype.getName = function(){ return this.name;}Parent.prototype.obj = {a:1};var Child = function(name){ Parent.apply(this,arguments);}Child.prototype = new Parent();var parent = new Parent('myparent');var child = new Child('mychild');console.log(parent.getName()); //myparentconsole.log(child.getName()); //mychild
上面这种方法在子类构造函数中通过apply调用父类的构造函数来进行相同的初始化工作,这样不管父类中做了多少初始化工作,子类也可以执行同样的初始化工作。但是上面这种实现还存在一个问题,父类构造函数被执行了两次,一次是在子类构造函数中,一次在赋值子类原型时Child.prototype = new Parent() ;,这是很多余的,所以我们还需要做一个改进:
var Parent = function(name){ this.name = name || 'parent' ;} ;Parent.prototype.getName = function(){ return this.name ;} ;Parent.prototype.obj = {a : 1} ;var Child = function(name){ Parent.apply(this,arguments) ;} ;Child.prototype = Parent.prototype ;var parent = new Parent('myParent') ;var child = new Child('myChild') ;console.log(parent.getName()) ; //myParentconsole.log(child.getName()) ; //myChild
这样我们就只需要在子类构造函数中执行一次父类的构造函数,同时又可以继承父类原型中的属性,这也比较符合原型的初衷,就是把需要复用的内容放在原型中,我们也只是继承了原型中可复用的内容。
上面借用构造函数模式最后改进的版本还是存在问题,它把父类的原型直接赋值给子类的原型,这就会造成一个问题,就是如果对子类的原型做了修改,那么这个修改同时也会影响到父类的原型,进而影响父类对象。
var Parent = function(name){ this.name = name || 'parent' ;} ;Parent.prototype.getName = function(){ return this.name ;} ;Parent.prototype.obj = {a : 1} ;var Child = function(name){ Parent.apply(this,arguments) ;} ;var F = new Function(){} ;F.prototype = Parent.prototype ;Child.prototype = new F() ;var parent = new Parent('myParent') ;var child = new Child('myChild') ;console.log(parent.getName()) ; //myParentconsole.log(child.getName()) ; //myChild
很容易可以看出,通过在父类原型和子类原型之间加入一个临时的构造函数F,切断了子类原型和父类原型之间的联系,这样当子类原型做修改时就不会影响到父类原型。
根据上面的描述,只要子类对象中访问到的原型跟父类原型是同一个对象,那么就会出现上面这种情况,所以我们可以对父类原型进行拷贝然后再赋值给子类原型,这样当子类修改原型中的属性时就只是修改父类原型的一个拷贝,并不会影响到父类原型。具体实现如下:
递归拷贝式继承
function deepClone(initalObj,finalObj){ var obj = finalObj || {}; for(var i in initalObj){ //逐个取对象的属性 if(initalObj.hasOwnProperty(i)){ //判断不是通过原型链继承的属性 if(typeof initalObj[i] === 'object'){ //属性对应的value是object(这里可能是数组) obj[i] = (initalObj[i].constructor === Array) ? [] : {}; //typeof数组出来的也是“object”,可以用.constructor来判断是否为数组 arguments.callee(initalObj[i],obj[i]); }else{ obj[i] = initalObj[i]; } } } return obj;}var str = {};var obj = {a: {a: 'hello',b: 21}};deepClone(obj,str);console.log(str.a);
var Parent = function(name){ this.name = name || 'parent' ;} ;Parent.prototype.getName = function(){ return this.name ;} ;Parent.prototype.obj = {a : 1} ;var Child = function(name){ Parent.apply(this,arguments) ;} ;Child.prototype = deepClone(Parent.prototype) ;Child.prototype.constructor = Child ;var parent = new Parent('myParent') ;var child = new Child('myChild') ;console.log(parent.getName()) ; //myParentconsole.log(child.getName()) ; //myChild
阅读全文
0 0
- JavaScript实现继承的方式优化
- javascript实现继承的方式
- javascript实现继承的方式
- javascript实现继承的方式
- Javascript继承的实现方式
- javascript 继承实现方式
- javascript 继承实现方式
- Javascript继承实现方式
- javascript【继承实现方式 】
- javascript实现继承的多种方式
- javascript中的继承的实现方式
- Javascript 实现继承的三种方式
- JavaScript实现继承的五种方式
- javascript 实现继承的几种方式
- Javascript实现继承的6种方式
- javascript实现继承的几种方式
- javaScript实现继承的5种方式
- javascript实现继承的几种方式
- 字符串排列组合问题&递归算法(2)
- Java Annotation最佳入门实践
- 2017暑假七林集训day11
- js 数组和对象的深度拷贝
- js中eval函数对json格式数据的处理
- JavaScript实现继承的方式优化
- 基本图像操作
- Android面试必会
- JVM垃圾回收根搜索的三种算法
- 1.注解@pathvariable的用法
- 初识NodeJS服务端开发(Express+MySQL)
- Android中将图片保存到数据库中
- 2 SOLR2:SOLR整合TOMCAT+导入数据并创建索引+solrj操作
- 库存分析与控制课后习题