返回顶部
首页 > 资讯 > 前端开发 > JavaScript >深入JS继承
  • 361
分享到

深入JS继承

2024-04-02 19:04:59 361人浏览 独家记忆
摘要

目录前言准备总结继承的n种方式原型式继承原型链式继承借用构造函数(类式继承)组合继承寄生组合式继承结束语前言 对于灵活的js而言,继承相比于java等语言,继承实现方式可谓百花齐放。

前言

对于灵活的js而言,继承相比于java等语言,继承实现方式可谓百花齐放。方式的多样就意味着知识点繁多,当然也是面试时绕不开的点。撇开es6 class不谈,传统的继承方式你知道几种?每种实现原理是什么,优劣点能谈谈吗。这里就结合具体例子,按照渐进式的思路来看看继承的发展。

准备

谈到js继承之前先回顾下js 实例化对象的实现方式。

构造函数是指可以通过new 来实例化对象的函数,目的就是为了复用,避免每次都手动声明对象实例。

new 简单实现如下:


function my_new(func){
    var obj = {}
    obj._proto_ = func.prototype // 修改原型链指向,拼接至func原型链
    func.call(obj) // 实例属性赋值
    return obj
}

由上可以看出,通过构造函数调用,可以将实例属性赋值到目标对象上。

如此可以推想,子类中调用父类构造函数同样可以达到继承的目的。

这就提供了js继承的一种思路,即通过构造函数调用。

至于原型属性,就是通过修改原型指向,来实现原型属性的共享。

那么继承时同样也可以通过该方式进行。

总结

基于构造函数和原型链两种特性,结合js语言的灵活性。

继承的实现方式虽然繁多万变也不离其宗

继承的n种方式

原型式继承

定义:这种继承借助原型并基于已有的对象创建新对象,同时还不用创建自定义类型的方式称为原型式继承。

直接看代码更清晰:


function createObj(o) {
  function F() { }
  F.prototype = o;
  return new F();
}
var parent = {
  name: 'trigkit4',
  arr: ['brother', 'sister', 'baba']
};
var child1 = createObj(parent);

该方式表面上看基于对象创建,不需要构造函数(当然实际构造函数被封装起来罢了)。只借助了原型对象,所以名称为原型式继承。

缺点:

比较明显优良者

无法复用该继承,每个子类的实例,都要走完整的createObj流程。

对于子类对象

因为构造函数封装createObj中,对其而言,没有构造函数。由此造成无法初始化时传参。
补充:其中 createObj 就是我们ES6中常用的Object.create(),不过Object.create进行了完善,允许额外参数来完善了。

解决思路:

既然提到没有构造函数导致了问题,那么大胆猜测,更进一步就是涉及了构造函数的原型链继承了。

原型链式继承

定义:为了让子类继承父类的属性(也包括方法),首先需要定义一个构造函数。然后,将父类的新实例赋值给构造函数的原型。


function Parent() {
  this.name = 'mike';
}
function Child() {
  this.age = 12;
}
Child.prototype = new Parent();
child.prototype.contructor = child // 原型属性被覆盖,所以要修正回来。
var child1 = new Child();

也就是直接修改子类的原型对象指父构造函数的实例,这样把父类的实例属性和原型属性都挂到自己原型链上。

缺点

Child.prototype = new Parent() ,那么子函数自身的原型属性就被覆盖了,如果需要就要在后面补充。

子对象实例化时,无法向父类构造函数传递参数。
例如在new Child()执行的时候,想要去覆盖name,只能在Child.prototype = new Parent()时。 是我们在new Child()的时候统一传参初始化是更常规需求。

解决思路

如何在子类初始化时,调用父类构造函数。结合前面的基础,答案也呼之欲出。

借用构造函数(类式继承)

类式继承:是在子类型构造函数的内部调用超类型的构造函数。

思路比较清晰,由问题驱动。

既然原型链式子类不能向父类传参的问题,那么在子类初始化是调用父类不就满足目的了。

示例如下:


function Parent(age) {
  this.name = ['mike', 'jack', 'smith'];
  this.age = age;
}
Parent.prototype.run = function () {
  return this.name + ' are both' + this.age;
};
function Child(age) {
  // 调用父类
  Parent.call(this, age);
}
var child1 = new Child(21);

这样满足了初始化时传参的需求,但是问题也比较明显。


child1.run //undefined

问题

父类原型属性丢失

父类初始化只继承了示例属性,原型属性在子类的原型链上丢失

解决思路

丢失的原因在于原型链没有修改指向,那么修改下指向不就完了。

组合继承

定义:使用原型链实现对原型属性和方法的继承,而通过借用构造函数来实现对实例属性的继承

示例:


function Parent(age) {
  this.name = ['mike', 'jack', 'smith'];
  this.age = age;
}
Parent.prototype.run = function () {
  return this.name + ' are both' + this.age;
};
function Child(age) {
  // 调用父类构造函数
  Parent.call(this, age);
}
Child.prototype = new Parent();//原型属性继承
Child.prototype.contructor = Child
var child1 = new Child(21);

这样问题就避免了:


child1.run() // "mike,jack,smith are both21"

问题

功能满足之后,就该关注性能了。这种继承方式问题在于父类构造函数执行了两次。

分别是:


function Child(age) {
  // 调用父类构造函数,第二次
  Parent.call(this, age);
}
Child.prototype = new Parent();//修改原型链指向,第一次

解决思路

解决自然是取消一次构造函数调用,要取消自然要分析这两次执行,功能上是否有重复。

第一次同样继承了实例和原型属性,第二次执行同样继承了父类的实例属性。

因此第二次满足对父类传参的不可获取性,因此只能思考能否第一次不调用父类构造函数,只继承原型属性。

答案自然是能,前面原型式继承就是这个思路。

寄生组合式继承

顾名思义,寄生指的是将继承原型属性的方法封装在特定方法中,组合的是将构造函数继承组合起来,补充原型式继承的不足。

饶了点,直接看:


function createObj(o) {
  function F() { }
  F.prototype = o;
  return new F();
}
//继承原型属性 即原型式继承
function create(parent, child) { 
  var f = createObj(parent.prototype);//获取原型对象
  child.prototype = f
  child.prototype.constructor = child;//增强对象原型,即保持原有constructor指向
}

function Parent(name) {
  this.name = name;
  this.arr = ['brother', 'sister', 'parents'];
}
Parent.prototype.run = function () {
  return this.name;
};
function Child(name, age) {
  // 示例属性
  Parent.call(this, name);
  this.age = age;
}
// 原型属性继承寄生于该方法中
create(Parent.prototype,Child);
var child1 = new Child('trigkit4', 21);

这样沿着发现问题解决问题的思路直到相对完善的继承方式。至于ES的方式本篇就不涉及了。

结束语

唯有厚积,才能薄发,想要心仪的offer,就得准备充裕,夯实基础,切忌似是而非,道理我都懂就是答得不完全,这样跟不懂差别也不太大。不算新的日子里立个flag,每周三个知识点回顾。要去相信,你若盛开蝴蝶自来。

以上就是深入JS继承的详细内容,更多关于深入JS继承的资料请关注编程网其它相关文章!

--结束END--

本文标题: 深入JS继承

本文链接: https://lsjlt.com/news/125156.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 深入JS继承
    目录前言准备总结继承的n种方式原型式继承原型链式继承借用构造函数(类式继承)组合继承寄生组合式继承结束语前言 对于灵活的js而言,继承相比于java等语言,继承实现方式可谓百花齐放。...
    99+
    2024-04-02
  • 深入理解C#之继承
    目录C#之继承能够阻止某个类被其他类继承吗?那么我们该如何获得基类的构造函数和自身的构造函数呢?都定义有构造函数时,那么执行顺序会怎样呢?总结C#之继承 继承、封装和多态是面向对象编...
    99+
    2024-04-02
  • 深入了解Python的继承
    目录面向对象三大特性:1、单继承1.1 继承的概念、语法和特点1)、继承的语法2)、专业术语总结面向对象三大特性: 封装 根据 职责 将 属性 和 方法 封装 到一个抽象的...
    99+
    2024-04-02
  • 深入浅析JAVA的继承
    本篇文章给大家分享的是有关深入浅析JAVA的继承,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。继承(inheritance)是Java OOP中一个非常重要的概念。继承是在复用...
    99+
    2023-05-31
    java 继承 ava
  • 深入浅析python继承问题
    有如下的代码: class p1: def __init__(self,a,b): print("init in p1") self.a1=a self.b1=b self.f1() def f1(se...
    99+
    2022-06-04
    python
  • Java由浅入深刨析继承
    目录继承继承的介绍生活中的继承继承的好处继承的格式继承的demo子类不能继承的内容super 与 this 关键字构造器不能被继承final修饰的类不能被继承方法重写介绍使用场景与案...
    99+
    2024-04-02
  • Java由浅入深讲解继承下
    目录1.代码块初始化2.protected关键字3.final关键字4.Java中允许的继承方式本篇紧接上一篇内容继续,还是从继承里的细节开始 1.代码块初始化 关于代码块的定义和使...
    99+
    2024-04-02
  • Java由浅入深讲解继承上
    目录1.什么是继承2.继承的细节2.1super关键字2.2子类的构造方法2.3super和this区别继承同样是面向对象程序的特点 1.什么是继承 所谓继承就是抽取类的共性,进而实...
    99+
    2024-04-02
  • C++继承的赋值转换与菱形虚拟继承深入详解
    目录一、继承的概念及定义1.1、继承的概念1.2、继承的定义二、基类和派生类对象赋值转换三、继承中的作用域3.1、继承同名成员处理方式3.2、继承同名静态成员处理方式3.3、继承与友...
    99+
    2022-11-13
    C++继承的赋值转换 C++菱形虚拟继承
  • C++中的继承模式深入详解
    前言 继承是OOP设计中的重要概念。在C++语言中,派生类继承基类有三种继承方式:私有继承(private)、保护继承(protected)和公有继承(public)。 一、继承...
    99+
    2024-04-02
  • 深入浅析kotlin中的继承与类
    深入浅析kotlin中的继承与类?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。kotlin基础教程之类和继承类声明使用class关键字声明类,查看其声明格式:: modifie...
    99+
    2023-05-31
    kotlin 继承
  • 深入浅析Java中的异常继承
    这篇文章给大家介绍深入浅析Java中的异常继承,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java异常类层次结构图: 异常的英文单词是exception,字面翻译就是“意外、例外”的意思,也就是非正常情况。...
    99+
    2023-05-31
    java 异常继承 ava
  • 深入解析Python多继承的机制
    深入探讨Python中的多继承机制引言:在Python中,多继承是一种强大而灵活的机制。通过多继承,我们可以在一个类中同时集成多个父类的属性和方法,大大增强了类的功能。多继承的基本概念多继承,即一个子类可以同时继承多个父类的属性和方法。这种...
    99+
    2023-12-30
    Python 机制 多继承
  • js中ES6继承和ES5继承之间的差别
    目录继承方式ES5prototype继承ES6class继承二者区别ES5prototype继承内部实现方式ES6class继承内部实现方式扩展继承方式 ES5 prototype ...
    99+
    2024-04-02
  • Java由浅入深带你精通继承super
    目录什么是继承背景super关键字protected 关键字final 关键字什么是继承 面向对象的特征: 封装:不必要公开的数据成员和方法,使用private关键字进行修饰。意义:...
    99+
    2024-04-02
  • Flask模板继承深入理解与应用
    目录什么叫模板继承呢模板页完整代码什么叫模板继承呢 在我的理解就是:在前端页面中肯定有很多页面中有很多相同的地方,比如页面顶部的导航栏,底部的页脚等部分,这时候如果每一个页面都重写一...
    99+
    2024-04-02
  • C++深入探究不同的继承体系
    目录单继承多继承菱形继承概念存在的问题解决方案菱形虚拟继承虚拟继承什么是虚拟继承内存层面理解虚拟继承虚拟继承和普通单继承的区别虚拟继承+菱形继承声明: 本文的测试环境为Windows...
    99+
    2024-04-02
  • JavaScript 继承的方方面面:深入浅出
    引言 继承是面向对象编程中的一项基本机制,它允许创建派生类,从基类继承属性和方法。在 JavaScript 中,有两种主要类型的继承:通过原型链和 ES6 中引入的类。本文将深入探索 JavaScript 继承的方方面面,从基础概念到高...
    99+
    2024-02-15
    JavaScript 继承 ES6 原型链
  • JS怎样实现继承
    这篇文章将为大家详细讲解有关JS怎样继承,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。原型继承得靠原型来实现,当然原型不是这篇文章的重点,我们来复习一下即可...
    99+
    2024-04-02
  • C++深入探究继承的概念与使用
    目录1、概念及定义1.1 概念1.2 定义2、class与struct的区别3、赋值兼容规则4、继承中的作用域问题5、派生类(子类)的默认成员函数5.1 构造函数5.2 拷贝构造函数...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作