关于JS继承

关于继承这个概念,是由面向对象衍生出来的。而JS身为一种基于面向对象而设计的语言,它和正统的面向对象语言又是有差别的。

面向对象语言

身为面向对象语言需要对开发者提供四种能力:

①:封装 - 把相关的信息(无论数据或方法)存储在对象中的能力;

②:聚集 - 把一个对象存储在另一个对象内的能力;

③:多态 - 编写能以多种方法运行的函数或方法的能力;

④:继承 - 有另一个或多个类得来类的属性和方法的能力;

虽然JS都具备这种能力,然而它却和正统的面向对象语言有差异,具体表现主要在第③、④点。

多态

多态主要有两种实现形式:

①:重载(编译时多态性) - 方法名相同,形参个数或类型不同;

②:重写(运行时多态性) - 在继承中,子类重写父类的方法;

这两种形式上JS和java/C#是表现形式是不一样的。

重载:

// java代码示例
public void sum(int n, float m){ }
public void sum(int n, float m, int t){ }
public void sum(int n, float m, String t){ }

在java中这sum方法会分别在不同情况下调用。

// js代码示例
function sum(n,m){ }
function sum(n,m,t){ }

在JS中sum方法在前面定义的会被后面定义的覆盖,因此JS用相同方法处理不用情况只能在sum方法内部对数据做识别判断了。

重写:

//java代码示例
public class Main { public static void main(String[] args) {
Student a = new Student();
a.say();
}
} class person{
public void say(){
System.out.println("11111");
}
} class Student extends person{
public void say(){
System.out.println("22222");
}
}

这里Student继承了person,对say方法进行了重写,虽然重写了say方法,但是父类方法不会被改变。

//js代码示例
function A(m){
this.m = m;
}
A.prototype.getM = function(){
console.log(100,this.m)
} function B(n){
this.n = n;
}
B.prototype = new A(100);
B.prototype.getN = function(){
console.log(this.n)
} let b = new B(200);
b.getM(); //打印 100,100 b.getM = function(){
console.log(1)
}
b.getM(); //打印 1
A.prototype.getM(); //打印 100,undefined b.__proto__.__proto__.getM = function(){
console.log(0,this.m)
};
b.getM(); //打印 1
A.prototype.getM(); //打印 0,undefined

而JS对继承的父类的重写就有点不一样了,JS是基于原型链的继承。调用方法或对象时会在原型链中追溯,而当在实例b中添加了getM方法时,其实是拦截了原型链上的getM,因此这种拦截不会改动到A.prototype.getM,然而A.prototype.getM又不是绝对的私密的,在示例中可以看到利用b.__proto__.__proto__.getM重写getM是会导致A.prototype.getM也被重写的,所以js原型链继承中是没有完整意义上的私密属性的。

继承

和其他正统的面向对象语言不一样,其他语言的继承一般都是拷贝继承,也就是子类会把父类中的方法和属性拷贝的子类中实现继承,而JS的继承是把父类的原型放在子类的原型链上,利用原型链的查找机制实现继承的。

而且JS的继承方式也是很多的,下面列举一下比较常见的继承方式。

一、原型链继承
function A(m){
this.m = m;
}
A.prototype.getM = function(){
console.log(this.m)
} function B(n){
this.n = n;
}
B.prototype = new A(100);
B.prototype.getN = function(){
console.log(this.n)
} let b = new B(200);
b.getN(); //输出 200
b.getM(); //输出 100

特点:

1、非常纯粹的继承关系,实例是子类的实例,也是父类的实例;

2、父类新增属性或其原型新增属性,子类都能访问;

缺点:

1、子类能重写父类方法,如:b.__proto__.__proto__.getM = null,可以改写父类方法;

2、父类的私有属性和公有属性都变成子类的公有属性;

3、继承时方法B原型上原有的方法或属性会丢失,如:constructor构造函数;(解决:B.prototype = B);

4、子类创建实例时无法向父类传参;

5、无法实现多继承;

二、构造函数继承
function A(m, n){
this.m=m;
this.n=n;
}
function B(m, n, t){
A.call(this, m, n);
this.t=t;
}
var b = new B(1, 2, 3);
console.log(b.m, b.n, b.t);

特点:

1、能在子类创建实例时向父类传递参数;

2、可以实现多继承;

缺点:

1、只能继承父类的属性(不继承原型链);

2、创建的实例携带着父类属性,臃肿;

三、寄生组合继承
function B(n){
this.n = n;
}
B.prototype.getB = function(name){
console.log(this.n, name);
} function A(m){
B.call(this,200)
this.m = m;
} // 我是旧时代的残党,新时代没有适合承载我的船
// (function(){
// // 创建一个没有实例方法的类
// var Super = function(){};
// Super.prototype = B.prototype;
// //将实例作为子类的原型
// A.prototype = new Super();
// })(); // ECMAScript 5 通过新增 Object.create()方法规范化了原型式继承。
A.prototype = Object.create(B.prototype); A.prototype.constructor = A;
A.prototype.getA = function(name){
console.log(this.m, name)
} var aa = new A(100);
aa.getA("A");
aa.getB("B");

特点:

1、可以继承父类的属性及其原型上的属性;

2、既是子类的实例,也是父类的实例;

3、不存在引用属性共享问题;

4、父类可接收传参;

关于JS继承的更多相关文章

  1. js继承

    js继承有5种实现方式: 继承第一种方式:对象冒充 function Parent(username){ this.username = username; this.hello = function ...

  2. js继承之call,apply和prototype随谈

    在js中,call,apply和prototype都可以实现对象的继承,下面我们看一个例子: function FatherObj1() { this.sayhello = "I am jo ...

  3. js继承精益求精之寄生式组合继承

    一.混合/组合继承的不足 上一篇JS继承终于混合继承,认真思考一下,发现其还是有不足之处的: 空间上的冗余:在使用原型链的方法继承父类的原型属性(Animal.prototype)的同时,也在子类的原 ...

  4. 老生常谈--Js继承小结

    一直以来,对Js的继承有所认识,但是认识不全面,没什么深刻印象.于是,经常性的浪费很多时间重新看博文学习继承,今天工作不是特别忙,有幸看到了http://www.slideshare.net/stoy ...

  5. Js继承小结

    Js继承小结 一直以来,对Js的继承有所认识,但是认识不全面,没什么深刻印象.于是,经常性的浪费很多时间重新看博文学习继承,今天工作不是特别忙,有幸看到了http://www.slideshare.n ...

  6. js继承实现

    JS实现继承可以分为:对象冒充和原型链继承 其中对象冒充又包括:临时变量,call 和 apply 临时变量方法: function Person(name,sex){ this.name = nam ...

  7. js继承之借用构造函数继承

    我的上一篇文章介绍了,原型链继承模式.但是单纯的原型链模式并不能很好地实现继承. 一.原型链的缺点 1.1 单纯的原型链继承最大的一个缺点,来自于原型中包含引用类型的值. 本来,我们没有通过原型链实现 ...

  8. js继承之原型链继承

    面向对象编程都会涉及到继承这个概念,JS中实现继承的方式主要是通过原型链的方法. 一.构造函数.原型与实例之间的关系 每创建一个函数,该函数就会自动带有一个 prototype 属性.该属性是个指针, ...

  9. js继承的常用方法

    写在前面的话:这篇博客不适合对面向对象一无所知的人,如果你连_proto_.prototype...都不是很了解的话,建议还是先去了解一下JavaScript面向对象的基础知识,毕竟胖子不是一口吃成的 ...

  10. JS--我发现,原来你是这样的JS:面向对象编程OOP[3]--(JS继承)

    一.面向对象编程(继承) 这篇博客是面向对象编程的第三篇,JS继承.继承顾名思义,就是获取父辈的各种"财产"(属性和方法). 怎么实现继承? 我们的JavaScript比较特别了, ...

随机推荐

  1. 1.IO流

    1.组成部分 File.FileInputStream.FileOutStream.FileWriter.FileReader 2.File pathSeparatorChar ; separator ...

  2. golang中数组指针与指针数组的区别实现

      指针数组和数组的指针,指的是两个不同的东西. 指针数组是有指针组成的数组,数组的指针是一个数组的指针. package main import "fmt" const MAX ...

  3. logrotate 日志切割

    logrotate是一个日志文件管理工具.用于分割日志文件,删除旧的日志文件,并创建新的日志文件,起到"转储"作用. 配置文件 Linux系统默认安装logrotate工具,它默认 ...

  4. python23day

    内容回顾 面向对象 类:是具有相同属性和相似功能的一类事物 对象/实例:具体的,一类可以有多个对象 实例化 练习 # 定义一个圆形类,半径是这个圆的属性,实例化一个半径为5的圆形,一个半径为10的圆形 ...

  5. HTML 基础1

    HTML 超文本标记语言 文件后缀html,htm 标签成对出现:开始标签--结束标签 元素内容位于开始标签--结束标签之间(可以有空内容) 空元素<a/> 大小写不敏感 元素,属性 &l ...

  6. python 小兵之小技巧

    用for循环打印数字从1开始 for a in range(1,num+1): 用split切割字符串可以用索引选择部分 int(el.split("_")[1]) range 第 ...

  7. Win11怎么启动任务管理器?Win11启动任务管理器的四种方法

    Win11怎么启动任务管理器?小编为大家带来了Win11启动任务管理器的四种方法,感兴趣的朋友一起看看吧 任务管理器是Windows系统中一项非常实用的功能.不过在最新版Win11中,右击任务栏启动任 ...

  8. 记一次.net core 异步线程设置超时时间

    前言: 刷帖子看到一篇 Go 记录一次groutine通信与context控制 看了一下需求背景,挺有意思的,琢磨了下.net core下的实现 需求背景: 项目中需要定期执行任务A来做一些辅助的工作 ...

  9. JDK及JRE简介

    JDK(Java Development Kit)是Sun Microsystems针对Java开发员的产品.自从Java推出以来,JDK已经成为使用最广泛的Java SDK.JDK 是整个Java的 ...

  10. 抓包神器 tcpdump 使用介绍 (转)

    tcpdump 命令使用简介 简单介绍 tcpdump 是一款强大的网络抓包工具,运行在 linux 平台上.熟悉 tcpdump 的使用能够帮助你分析.调试网络数据. 要想使用很好地掌握 tcpdu ...