JS--我发现,原来你是这样的JS:面向对象编程OOP[3]--(JS继承)
一、面向对象编程(继承)
这篇博客是面向对象编程的第三篇,JS继承。继承顾名思义,就是获取父辈的各种"财产"(属性和方法)。
怎么实现继承?
我们的JavaScript比较特别了,主要通过原型链实现继承的。
下面介绍各种实现继承的方式:原型链继承,借用构造函数,组合继承,原型式继承,寄生式继承,寄生组合式继承。
二、实现继承方式
1.原型链方式
原型我们都知道,每个构造函数都有一个原型对象(prototype),用于存放共享的属性方法。
原型链继承原理:我们要继承一个父类,那就把这个子类的原型对象指向父类的实例就行了。
a.代码:
//创建父类构造函数
function Father(){
this.fristName = "Father";
}
//给父类的原型对象添加方法
Father.prototype.getFatherName = function(){
return this.fristName ;
}
//创建子类构造函数
function Son(){
this.name = 'jack';
}
//重点这行:把子类的原型对象指向父类实例实现继承
Son.prototype = new Father();
//我们还可以为子类添加原型方法
Son.prototype.getName = function(){
return this.name;
}
//测试一下,创建一个子类的实例
var s1 = Son();
s1.getFaterName(); //Father 继承了爸爸的名字了!
b.一些特别注意的地方:
1.上面我们看到为子类原型添加方法getName,它在子类原型对象指向父类实例后,也一定要在这句话后面,如果在前面的话,这方法会被覆盖的。因为子类原型对象指向父类实例时相当于重写子类原型对象。
2.为子类原型添加方法式不能用字面量的方式,这样会重写已经继承类父类的原型对象。
c.原型链继承的问题:
很明显的,继承的属性方法都在子类的原型对象里面,那么里面的属性方法都是共享的,这明显不是符合我们平常要求。
2.借用构造函数
借?是的,借一下父类的构造函数,怎么借?那就使用call或者apply方法吧。
借用构造函数原理: 就在子类创建构造函数时内部调用父类的构造函数。
代码演示:
//父类(父类是相对来说的,大家注意一下噢)
function Father(){
this.colors = ['red','green'];
}
//子类
function Son(){
//重点这行:内部调用父类构造函数,this是指向未来创建Son的实例
Father.call(this);
}
//测试一下,继承了父类的属性了,并且属性不会共享噢
var s1 = new Son();
s1.colors.push('blue');
console.log(s1.colors); ['red','green','blue']
//s2实例的colors不会被影响
var s2 = new Son();
console.log(s2.colors); ['red','green']
这里的借用构造函数可以实现属性和方法不会被共享,因为属性在构造函数中,而不是在原型对象中。但是这也说明这种方式没办法共享方法的。
3.组合继承(最常用)
组合继承=原型链继承+借用构造函数继承,没错就结合两种方法就好了。
组合继承原理:实例共享的属性方法:我就原型链继承;实例私用的属性方法:我就借用构造函数(结合两者的优点)
代码演示:
//父类
function Father(name){
this.name = name;
this.colors = ['red','green'];
}
//父类原型对象
Father.pototype.sayName = function(){
console.log(this.name);
}
//子类
function Son(name,age){
//借用父类构造函数
Father.call(this,name);
this.age = age;
}
//子类原型指向父类实例,原型链继承
Son.prototype = new Father();
Son.prototype.constructor = Son; //让子类的原型只向子类构造函数
Son.prototype.sayAge = function(){
console.log(this.age);
}
//创建实例对象
var s1 = new Son('Mc ry',22);
var s2 = new Son('yy',20);
//继承了父类方法,sayName是共享的
s1.sayName(); //mc ry
s2.sayName(); //yy
//继承colors,每个实例私有
s1.colors.push('blue');
console.log(s1.colors); ['red','green','blue']
console.log(s2.colors); ['red','green']
可能有的疑惑:在原型链继承那里,子类原型对象指向类父类的实例,应该继承了所有的属性和方法啊,那应该都是共享的,但为什么colors,name属性不会共享呢?
原因:在调用借用构造函数时,属性在子类新实例中创建了,也就是在子类实例中已经有的父类属性就不用继续到原型对象中查找了,也就是屏蔽了,所以不会共享了。
4.原型式继承
这是道格拉斯·克罗克福提出的一个方式,他提出一个函数object() 来简单实现不用构造函数的继承
代码演示:
//object函数,传入一个对象
function object(o){
//创建一个空的构造函数
function F(){};
//这个空构造函数的原型对象指向o这个对象
F.prototype = o ;
//返回F的实例对象
return new F();
}
认真读这段代码可以知道,这个函数最终返回一个对象,这个对象拥有传入对象o的全部属性和方法。从而实现了继承。
//一个对象
var person = {
name : 'ry',
sayName : function(){
console.log(this.name);
}
}
//children对象继承了person对象所有的属性方法
var children = object(person);
原型式继承方式的出现,可以不用创建构造函数都能继承另一个对象的属性方法。但是很明显,所用属性都是共享的。
ES5中有个object.create()方法的作用和上面的object()一样的。
5.寄生式的继承
寄生式其实和利用了原型式,都要使用到object函数,但不同的是寄生式中新的对象能够添加自己的方法。
function creatAnother(o){
//调用object继承o
var obj = object(o);
//还可以添加自己想要的方法
obj.sayHi = function (){
console.log('hi');
}
//返回这个对象
retrun obj;
}
//新的对象继承了person
var anotherPerson = creatAnother(person);
//继承后能使用person的方法
anotherPerson.sayHi(); //"hi"
6.寄生组合式继承(最理想)
上面组合式方式中虽然是最常用的,但有追求的还是会继续优化它。因为组合方式中也有不够好的地方:
一方面:我们可以看到调用了两次父类的构造函数。(一次是原型链继承中子类原型对象指向父类实例时,一次是借用构造函数中)
另一方面:就是上面疑惑,子类的原型中拥有父类已经有全部属性,但我们又要在调用子类构造函数时重写部分属性。
所以寄生组合方式就解决了上面,通过一个函数来代替组合方式中的原型链继承。
代码演示:
//主要是这个函数,实现子类原型对象只继承父类原型对象中的属性
function inheritPrototype(subType,superType){
//利用上面的Object函数,将父类的原型赋予prototype变量
var prototype = object(superType.prototype);
//将prototype的构造函数重新指向子类
prototype.constructor = subType;
//将prototype给sub的原型对象
subType.prototype = prototype;
}
//将前面的组合继承改写
//父类
function Father(name){
this.name = name;
this.colors = ['red','green'];
}
//父类原型对象
Father.pototype.sayName = function(){
console.log(this.name);
}
//子类
function Son(name,age){
//借用父类构造函数
Father.call(this,name);
this.age = age;
}
//(这行用inheritPrototype函数替换)子类原型指向父类实例,原型链继承
inheritPrototype(Son, Father);
Son.prototype.sayAge = function(){
console.log(this.age);
}
寄生组合式的继承方式是最理想的方式,它使得子类构造函数继承父类构造函数的属性,子类的原型对象继承父类原型对象的属性和方法。
三、小结
1.这次博客讲述了在js中是如何实现继承的,有很多中方式,但主要的是组合方式和寄生组合方式。继承后我们能够使用父类的属性和方法,增加了代码的重用性。
2.了解js继承作用:有助于我们去阅读一些框架的源码,可能本次代码有点难以理解,我打上了大量的注释,供大家一起阅读,还是那句话,多看几遍,其义自见。如果觉得有收获,就点个赞吧,关注我吧。
- 细心的朋友可能发现,我把文章的样式修改了,类似Markdown风格咯,看起来比较清爽,舒服。
同系列几篇:
第一篇:JavaScript--我发现,原来你是这样的JS(一)(初识)
JS--我发现,原来你是这样的JS:面向对象编程OOP[1]--(理解对象和对象属性类型)
JS--我发现,原来你是这样的JS:面向对象编程OOP[2]--(创建你的那个对象吧)
我发现,原来你是这样的JS全部文章汇总(点击此处)
本文出自博客园:http://www.cnblogs.com/Ry-yuan/
作者:Ry(渊源远愿)
欢迎转载,转载请标明出处,保留该字段。
JS--我发现,原来你是这样的JS:面向对象编程OOP[3]--(JS继承)的更多相关文章
- JavaScript--我发现,原来你是这样的JS:面向对象编程OOP[2]--(创建你的那个对象吧)
一.介绍 我们继续面向对象吧,这次是面向对象编程的第二篇,主要是讲创建对象的模式,希望大家能从博客中学到东西. 时间过得很快,还是不断的学习吧,为了自己的目标. 二.创建对象 1.前面的创建对象方式 ...
- JS--我发现,原来你是这样的JS:面向对象编程OOP[2]--(创建你的那个对象吧)
一.介绍 我们继续面向对象吧,这次是面向对象编程的第二篇,主要是讲创建对象的模式,希望大家能从博客中学到东西. 时间过得很快,还是不断的学习吧,为了自己的目标. 二.创建对象 1.前面的创建对象方式 ...
- 如何把js的代码写的更加容易维护(一)--面向对象编程
总是头疼javascript的代码写起来不可维护,那么看看下面的代码: (function (w, $) { var app = { init: function () { var me = this ...
- JavaScript--我发现,原来你是这样的JS:面向对象编程OOP[1]--(理解对象和对象属性类型)
一.介绍 老铁们,这次是JS的面向对象的编程OOP(虽然我没有对象,心累啊,但是可以自己创建啊,哈哈). JS高程里第六章的内容,这章内容在我看来是JS中很难理解的一部分.所以分成三篇博客来逐个理清. ...
- JS--我发现,原来你是这样的JS:面向对象编程OOP[1]--(理解对象和对象属性类型)
一.介绍 老铁们,这次是JS的面向对象的编程OOP(虽然我没有对象,心累啊,但是可以自己创建啊,哈哈). JS高程里第六章的内容,这章内容在我看来是JS中很难理解的一部分.所以分成三篇博客来逐个理清. ...
- JS 学习笔记 (七) 面向对象编程OOP
1.前言 创建对象有很多种方法,最常见的是字面量创建和new Object()创建.但是在需要创建多个相同结构的对象时,这两种方法就不太方便了. 如:创建多个学生信息的对象 let tom = { n ...
- JS面向对象编程(进阶理解)
JS 面向对象编程 如何创建JS对象 JSON语法声明对象(直接量声明对象) var obj = {}; 使用 Object 创建对象 var obj = new Object(); JS对象可以后期 ...
- JS - ES5与ES6面向对象编程
1.面向对象 1.1 两大编程思想 1.2 面向过程编程 POP(Process-oriented programming) 1.3 面向对象编程 OOP (Object Oriented Progr ...
- js面向对象编程:if中可以使用那些作为判断条件呢?
作者来源http://www.2cto.com/kf/201407/314978.html搬运 在所有编程语言中if是最长用的判断之一,但在js中到底哪些东西可以在if中式作为判断表达式呢? 例如如何 ...
随机推荐
- 手把手教你用npm发布一个包,详细教程
我们已经实现了路由的自动化构建,但是我们可以看到,一大串代码怼在里面.当然你也可以说,把它封装在一个JS文件里面,然后使用require('./autoRoute.js')给引入进来,那也行.但是,为 ...
- ZOJ 2002 Copying Books 二分 贪心
传送门:Zoj2002 题目大意:从左到右把一排数字k分,得到最小化最大份,如果有多组解,左边的尽量小. 思路:贪心+二分(参考青蛙过河). 方向:从右向左. 注意:有可能最小化时不够k分.如 ...
- Spring ——依赖注入配置一些知识点
依赖注入 依赖注入的原理与实现 依赖注入(DI)和依赖查找(Dependency Lookup)共同组成 控制反转(IoC).从原理的角度来说,依赖注入和控制反转是没 有不同的,可以看作是从两个角度来 ...
- win10 uwp 获得缩略图
有时候需要获得文件或视频的缩略图. 本文提供两个方法,用于获得文件的缩略图和截取视频指定时间的显示图片. 文件缩略图 如果有一个文件需要获得缩略图,可以使用 GetThumbnailAsync 或 G ...
- 最美时光第三方UWP源码公开
自己大概写了一个星期的成果.. 使用了官方最美时光app的UI和图片资源,并没有调用官方接口 https://files.cnblogs.com/files/loyieking/NiceCountDo ...
- 企业微信开发之发放企业红包(C#)
一.企业微信API 地址:http://work.weixin.qq.com/api/doc#11543 二.参数说明 1.发送企业红包 请求方式:POST(HTTPS)请求地址:https://ap ...
- 有序线性表(存储结构数组)--Java实现
/*有序数组:主要是为了提高查找的效率 *查找:无序数组--顺序查找,有序数组--折半查找 *其中插入比无序数组慢 * */ public class MyOrderedArray { private ...
- Nodejs.热部署方法
在开发中我们修改了一点代码后要去重启服务器才能看到结果,为了省去这个过程我们以往经常使用热部署代码的方法 下面是使用"supervisor"来达到热部署能力的方法: sudo np ...
- mybatis 参数为list时,校验list是否为空
校验objStatusList 是否为空 <if test="objStatusList != null and objStatusList.size() > 0 "& ...
- EF框架搭建小总结--CodeFirst模型优先
前言:之前在下总结编写了一篇 EF框架搭建小总结--ModelFirst模型优先 博文,看到一段时间内该博文的访问量蹭.蹭蹭.蹭蹭蹭...往上涨(实际也不是很多,嘿嘿),但是还是按捺不住内心的喜悦(蛮 ...