javascript 学习笔记(2) 原型链与继承

js里面是如何做到继承的呢?依靠一个东西叫原型链,所谓原型链,就是上一节所提到的原型组成的链条。(这到底是什么鬼)

之前说到每一个构造函数都有一个prototype属性,由构造函数new出来的实例都有一个指向prototype的内部指针。

那么,如果我们让一个构造函数的prototype等于另外一个对象(构造函数为A)的实例,那么会发生什么呢?

会发生的就是这个prototype对象中会有一个内部指针,指向A.prototype;

然后我们又可以将A.prototype等于其他的对象实例,就这么一而再,再而三,一条原型的链条就这么诞生了,这就是所谓的原型链。


function SuperType(){
  this.name = "Super";
}

SuperType.prototype.getSuperName(){
   return this.name;
}

function SubType(){
  this.subName="sub";
}

SubType.prototype = new SuperType();

SubType.prototype.getSubName = function(){
   return this.subName;
}

var obj = new SubType();
alert(obj.getSuperName());//Super


上面定义了两个类型,一个SuperType, 一个SubType,其中SubType的prototype指向的是SuperType的实例,而SuperType的实例

又包含了一个内部指针指向了SuperType.prototype,因此形成了原型链。因为原型链的存在,最终使得SubType的实例obj中可以调用

SuperType.prototype中定义的getSuperName方法(回顾一下上一节讲到的属性对象搜索过程,只是比上一节所讲的扩展了一下)。


原型链的问题:

仍旧是原型所固有的问题,就是共享,上一节中使用了属性初始化和方法初始化分离的方法解决了共享,但是这里又不同了,因为我们的

SubType的原型指向的是SuperType的实例,因此,很简单的,SuperType中的任何属性对象都会被SubType的对象所共享。然后,总会

有些不爽的事情发生,那肿么办呢?


借用构造函数:

啥叫借用构造函数,实际上就是运用了function.apply和function.call方法,强行修改了父类构造函数执行环境,例子如下:

function SuperType(name){...}

function SubType(name,type){
  this.type = type;
  SuperType.call(this,name);
}

SubType.prototype = new SuperType();
这里我们看到,虽然仍然是使用了SuperType的实例作为SubType的实例,但是我们在SubType的构造函数中借用了SuperType的构造函数来为

SubType的实例进行了初始化,实质上是让每一个SubType的实例在初始化之后都覆盖掉了自己原型中的属性,从而打破了属性的共享。这种继承

也被称之为组合式继承。这种继承方式很大程度上解决了js对象继承中所遇到的问题,但是,它仍然会有自己的问题,这个问题就是,SubType的

prototype中仍然存在着SuperType的属性,虽然被覆盖掉无法访问,但它毕竟是存在着的。随着原型链长度的增加,原型链中被不应该存在的属性

对象所占用的内存空间将会越来越大,因此,我们仍然需要对它进行改进。


原型式继承:

先把代码贴出来:

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

这种原型式的继承想法是要借助原型在已有的对象的基础上创建新的对象,我的理解是,这种继承方式,把继承从具体的对象类型上剥离出来,转

而将注意力着重在了原型对象的传递上。可能我这么说比较抽象,我们就来看看后面的这种继承方式:

寄生组合式继承:

function SuperType(name){
  this.name = name;
}

SuperType.prototype.getSuperName(){
   return this.name;
}

function SubType(name,type){
  this.type = type;
  SuperType.call(this,name);
}

SubType.prototype = object(SuperType.prototype)

SubType.prototype.getSubName = function(){
   return this.subName;
}
	

这里的object方法是借用了之前所提到的object方法,这里我们来思考一下,这种继承的好处是什么,其实区别就在于SubType赋值的时候,

没有new SuperType。我认为,这里就体现了我之前所提到的,将继承集中在了原型上,object方法的调用实际上就是把SuperType中我们

不需要的name属性从object返回的对象中剥夺了,相当于我们只是继承了SuperType.prototype中我们想要的方法,至于属性,留给借用构造

函数去完成,从而解决了上一种继承中冗余属性对象占用空间的问题。(即书中所讲的,组合式继承调用了两次SuperType的构造方法,而寄

生式组合继承只调用了一次)。


可能我讲的有点快,可能我讲的有点乱,希望有兴趣的朋友大家一起讨论!

下一节将说到闭包,这也是个重要概念啊!



Python网络爬虫与推荐算法新闻推荐平台:网络爬虫:通过Python实现新浪新闻的爬取,可爬取新闻页面上的标题、文本、图片、视频链接(保留排版) 推荐算法:权重衰减+标签推荐+区域推荐+热点推荐.zip项目工程资源经过严格测试可直接运行成功且功能正常的情况才上传,可轻松复刻,拿到资料包后可轻松复现出一样的项目,本人系统开发经验充足(全领域),有任何使用问题欢迎随时与我联系,我会及时为您解惑,提供帮助。 【资源内容】:包含完整源码+工程文件+说明(如有)等。答辩评审平均分达到96分,放心下载使用!可轻松复现,设计报告也可借鉴此项目,该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的。 【提供帮助】:有任何使用问题欢迎随时与我联系,我会及时解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 下载后请首先打开README文件(如有),项目工程可直接复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值