IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> 【JS继承】常见的7种继承方式 -> 正文阅读

[JavaScript知识库]【JS继承】常见的7种继承方式

一、原型式继承

核心:将父类的实例作为子类的原型。

SubType.prototype = new SuperType()

// 所有涉及到原型链继承的继承方式都要修改子类构造函数的指向,否则子类实例的构造函数会指向SuperType。

SubType.prototype.constructor = SubType;

优点:父类方法可以复用。

缺点:

  • 父类的引用属性会被所有子类实例共享

  • 子类构建实例时不能向父类传递参数

二、构造函数继承

核心:将父类构造函数的内容复制给了子类的构造函数。这是所有继承中唯一一个不涉及到prototype的继承。

SuperType.call(SubType);

优点:和原型链继承完全反过来

  • 父类的引用属性不会被共享

  • 子类构建实例时可以向父类传递参数

缺点:父类的方法不能复用,子类实例的方法每次都是单独创建的。

三、组合继承

核心:原型式继承和构造函数继承的组合,兼具了二者的优点。

function SuperType() {

? ?this.name = 'parent';

? ?this.arr = [1, 2, 3];

}

SuperType.prototype.say = function() {

? ?console.log('this is parent')

}

function SubType() {

? ?SuperType.call(this) // 第二次调用SuperType

}

SubType.prototype = new SuperType() // 第一次调用SuperType

优点:

  • 父类的方法可以被复用

  • 父类的引用属性不会被共享

  • 子类构建实例时可以向父类传递参数

缺点:调用了两次父类的构造函数,第一次给子类的原型添加了父类的name, arr属性,第二次又给子类的构造函数添加了父类的name, arr属性,从而覆盖了子类原型中的同名参数。这种被覆盖的情况造成了性能上的浪费。

四、原型式继承

核心:原型式继承的object方法本质上是对参数对象的一个浅复制。

优点:父类方法可以复用。

缺点:

  • 父类的引用属性会被所有子类实例共享

  • 子类构建实例时不能向父类传递参数

function object(o){

?function F(){}

?F.prototype = o;

?return new F();

}


var person = {

? ?name: "Nicholas",

? ?friends: ["Shelby", "Court", "Van"]

};


var anotherPerson = object(person);

anotherPerson.name = "Greg";

anotherPerson.friends.push("Rob");


var yetAnotherPerson = object(person);

yetAnotherPerson.name = "Linda";

yetAnotherPerson.friends.push("Barbie");

alert(person.friends); ? //"Shelby,Court,Van,Rob,Barbie"

ECMAScript 5 通过新增 Object.create()方法规范化了原型式继承。这个方法接收两个参数:一 个用作新对象原型的对象和(可选的)一个为新对象定义额外属性的对象。在传入一个参数的情况下, Object.create()与 object()方法的行为相同。——《JAVASCript高级编程》

所以上文中代码可以转变为:

var yetAnotherPerson = object(person); => var yetAnotherPerson = Object.create(person);

五、寄生式继承

核心:使用原型式继承获得一个目标对象的浅复制,然后增强这个浅复制的能力。

优缺点:仅提供一种思路,没什么优点。

function createAnother(original){

? ?var clone=object(original); ? ?//通过调用函数创建一个新对象

? ?clone.sayHi = function(){ ? ? ?//以某种方式来增强这个对象

? ? ? ?alert("hi");

? ?};

? ?return clone; ? ? ? ? ? ? ? ? ?//返回这个对象

}


var person = {

? ?name: "Nicholas",

? ?friends: ["Shelby", "Court", "Van"]

};


var anotherPerson = createAnother(person);

anotherPerson.sayHi(); //"hi"

六、寄生组合继承

刚才说到组合继承有一个会两次调用父类的构造函数造成浪费的缺点,寄生组合继承就可以解决这个问题。

function inheritPrototype(subType, superType){

? ?var prototype = object(superType.prototype); // 创建了父类原型的浅复制

? ?prototype.constructor = subType; ? ? ? ? ? ? // 修正原型的构造函数

? ?subType.prototype = prototype; ? ? ? ? ? ? ? // 将子类的原型替换为这个原型

}

function SuperType(name){

? ?this.name = name;

? ?this.colors = ["red", "blue", "green"];

}

SuperType.prototype.sayName = function(){

? ?alert(this.name);

};

function SubType(name, age){

? ?SuperType.call(this, name);

? ?this.age = age;

}

// 核心:因为是对父类原型的复制,所以不包含父类的构造函数,也就不会调用两次父类的构造函数造成浪费

inheritPrototype(SubType, SuperType);

SubType.prototype.sayAge = function(){

? ?alert(this.age);

}

优缺点:这是一种完美的继承方式。

七、ES6 Class extends

核心: ES6继承的结果和寄生组合继承相似,本质上,ES6继承是一种语法糖。但是,寄生组合继承是先创建子类实例this对象,然后再对其增强;而ES6先将父类实例对象的属性和方法,加到this上面(所以必须先调用super方法),然后再用子类的构造函数修改this。

class A {}

class B extends A {

?constructor() {

? ?super();

?}

}

ES6实现继承的具体原理:

class A {

}

class B {

}

Object.setPrototypeOf = function (obj, proto) {

?obj.__proto__ = proto;

?return obj;

}

// B 的实例继承 A 的实例
Object.setPrototypeOf(B.prototype, A.prototype);

// B 继承 A 的静态属性
Object.setPrototypeOf(B, A);

ES6继承与ES5继承的异同:

相同点:本质上ES6继承是ES5继承的语法糖。

不同点:

  • ES6继承中子类的构造函数的原型链指向父类的构造函数,ES5中使用的是构造函数复制,没有原型链指向。

  • ES6子类实例的构建,基于父类实例,ES5中不是。

  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-06-16 21:35:35  更:2022-06-16 21:38:21 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 10:49:56-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码