0

0

什么是JS的类继承?

星降

星降

发布时间:2025-08-30 11:17:01

|

662人浏览过

|

来源于php中文网

原创

JavaScript类继承通过extends实现子类复用父类属性方法,基于原型链但用class语法更直观清晰,提升代码可读性与维护性。

什么是js的类继承?

JavaScript中的类继承,简单来说,就是一种让一个“子类”能够从一个“父类”那里继承属性和方法的能力。它允许我们构建一个层级结构,让子类在拥有自己独特功能的同时,也能复用父类的通用行为,从而减少代码重复,提高可维护性。在ES6之后,我们有了

class
关键字,这让JS的继承语法变得更接近传统面向对象语言,虽然它的底层依然是基于原型链的。

解决方案

在我看来,理解JS的类继承,首先得从ES6引入的

class
语法说起。它提供了一种更清晰、更直观的方式来定义“蓝图”或“模板”,也就是我们常说的类。当我们说一个类“继承”另一个类时,其实就是通过
extends
关键字来建立这种关系。

想象一下,我们有一个通用的

Animal
类,它可能有一些所有动物都具备的属性,比如
name
age
,以及一些方法,比如
eat()

class Animal {
  constructor(name, age) {
    this.name = name;
    this.age = age;
  }

  eat() {
    console.log(`${this.name} is eating.`);
  }

  sleep() {
    console.log(`${this.name} is sleeping.`);
  }
}

现在,我们想创建一个

Dog
类,它不仅有动物的通用属性和方法,还有自己独特的东西,比如
bark()
方法。这时候,
Dog
就可以“继承”
Animal

class Dog extends Animal {
  constructor(name, age, breed) {
    // 关键点:在子类构造函数中,必须先调用super()。
    // super()调用的是父类(Animal)的构造函数,负责初始化父类部分的属性。
    super(name, age); 
    this.breed = breed; // Dog类特有的属性
  }

  bark() {
    console.log(`${this.name} barks: Woof! Woof!`);
  }

  // 我们可以覆盖父类的方法
  eat() {
    console.log(`${this.name} is happily munching on kibble.`);
  }

  // 也可以在覆盖的同时调用父类的方法
  // sayHello() {
  //   super.sayHello(); // 调用父类的sayHello方法
  //   console.log(`My breed is ${this.breed}.`);
  // }
}

const myDog = new Dog('Buddy', 3, 'Golden Retriever');
myDog.eat();    // 输出:Buddy is happily munching on kibble. (调用子类覆盖的方法)
myDog.sleep();  // 输出:Buddy is sleeping. (调用父类继承的方法)
myDog.bark();   // 输出:Buddy barks: Woof! Woof! (调用子类特有的方法)

console.log(myDog.name);  // Buddy
console.log(myDog.age);   // 3
console.log(myDog.breed); // Golden Retriever

这里面有几个非常核心的要点:

  1. extends
    关键字
    :它明确指出了
    Dog
    类是
    Animal
    类的子类。
  2. super()
    在子类构造函数中
    :这是必须的!在子类的
    constructor
    中,你不能在调用
    super()
    之前使用
    this
    super()
    的作用是调用父类的构造函数,确保父类部分的属性(比如
    name
    age
    )被正确初始化。如果子类没有定义自己的
    constructor
    ,JavaScript会默认生成一个调用
    super()
    的构造函数。
  3. 方法继承与覆盖:子类会自动继承父类的方法。如果子类定义了与父类同名的方法,那么子类的方法会“覆盖”父类的方法。你也可以在子类方法中通过
    super.methodName()
    来调用父类被覆盖的方法。

总的来说,JS的类继承提供了一种结构化的方式来组织代码,让我们的程序逻辑更加清晰,尤其是在处理具有层级关系的对象时,它显得尤为重要。

为什么ES6引入了Class语法,它解决了哪些痛点?

在我个人的开发经历中,ES6的

class
语法无疑是JavaScript发展史上的一个重要里程碑。在此之前,JavaScript的继承机制,也就是原型链继承,虽然强大且灵活,但对于许多从传统面向对象语言(如Java、C++)转过来的开发者来说,学习曲线确实比较陡峭。

ES6引入

class
语法,在我看来,主要解决了以下几个痛点:

  • 语法上的直观性与熟悉感:这是最直接的感受。
    class
    constructor
    extends
    super
    这些关键字,让JavaScript的代码看起来更像传统OOP语言。这大大降低了有其他语言背景的开发者入门JavaScript的门槛。不再需要手动操作
    prototype
    对象,不再需要复杂的
    Object.create
    call
    /
    apply
    组合来模拟继承,代码意图更加明确。
  • 提高了代码的可读性与可维护性:在ES5时代,一个复杂的继承链可能会涉及多层构造函数和原型链的修改,代码分散且不易理解。
    class
    语法将类的定义、继承关系、方法和属性都封装在一个结构中,使得代码结构更加紧凑和清晰。当项目规模变大,需要团队协作时,这种清晰的结构能显著提高沟通效率和维护便利性。
  • 统一了模块化开发模式:虽然
    class
    本身不是模块化方案,但它与ES6的模块(
    import
    /
    export
    )配合使用时,能更好地组织大型应用的代码结构。你可以清晰地导出和导入类,构建一个由各种组件和数据模型组成的应用程序。
  • 提供了更安全的
    super
    调用
    :在ES5中,调用父类构造函数或方法需要手动处理
    this
    的绑定问题,稍有不慎就可能出错。
    super
    关键字在
    class
    语法中提供了一种安全、明确的方式来引用父类,无论是调用父类构造函数(
    super()
    )还是父类方法(
    super.methodName()
    ),都由JavaScript引擎内部处理了
    this
    的正确指向,减少了出错的可能性。
  • 更好地支持静态方法和属性
    class
    语法直接提供了
    static
    关键字来定义静态方法和属性,这在ES5中也需要一些额外的技巧来实现,并且不如
    static
    关键字直观。

尽管

class
语法本质上是原型链继承的“语法糖”,但它在开发者体验和代码组织上的提升是巨大的。它并没有改变JavaScript的底层机制,而是提供了一个更高级、更友好的抽象层,让开发者能够以更熟悉的方式去思考和构建面向对象的代码。

在实际开发中,JS类继承有哪些常见的应用场景和最佳实践?

在日常的JavaScript开发中,类继承的应用非常广泛,尤其是在构建大型、复杂的应用时。但同时,我们也需要注意一些最佳实践,避免过度设计或引入不必要的复杂性。

常见的应用场景:

  • UI组件库与框架开发
    • 基础组件:例如,你可以有一个
      BaseComponent
      类,它定义了所有UI组件共有的生命周期方法(如
      render
      mount
      unmount
      )、状态管理逻辑或事件绑定机制。
    • 派生组件:然后,
      Button
      Input
      Modal
      等具体组件就可以继承
      BaseComponent
      ,并在此基础上添加自己的特定样式和行为。这在React等框架的类组件模式中尤为常见。
  • 错误处理
    • JavaScript内置的
      Error
      类是可继承的。我们可以创建自定义的错误类型,例如
      NetworkError
      AuthenticationError
      ValidationError
    • class NetworkError extends Error {
        constructor(message, status) {
          super(message);
          this.name = 'NetworkError';
          this.status = status;
        }
      }
      try {
        // 模拟网络请求失败
        throw new NetworkError('Failed to fetch data', 500);
      } catch (error) {
        if (error instanceof NetworkError) {
          console.error(`Status ${error.status}: ${error.message}`);
        } else {
          console.error('An unexpected error occurred:', error.message);
        }
      }
    • 这样做的好处是,在
      catch
      块中,我们可以通过
      instanceof
      操作符来精确判断错误的类型,从而执行不同的错误处理逻辑。
  • 数据模型与ORM/ODM
    • 前端或Node.js后端,如果你需要处理结构化的数据,可以定义一个
      BaseModel
      类,包含数据验证、序列化/反序列化、CRUD操作等通用方法。
    • 然后,
      User
      Product
      Order
      等具体的数据模型就可以继承
      BaseModel
      ,并添加各自特有的字段和业务逻辑。
  • 工具类与实用程序
    • 虽然很多工具函数可以直接作为纯函数存在,但在某些情况下,如果一组相关的工具函数需要共享一些状态或配置,或者它们之间存在明确的层级关系,那么类继承也可能是一个选项。

最佳实践:

  • 遵循Liskov替换原则(LSP):这是面向对象设计的一个核心原则,即子类应该能够替换掉父类,并且程序的行为不会因此改变。这意味着子类在扩展功能的同时,不应该破坏父类的原有契约或行为。
  • 避免过深的继承链:继承链过长(比如A -> B -> C -> D)会使代码变得非常复杂,难以理解和维护。一个对象的行为可能分散在多个父类中,追踪问题会变得困难。通常建议继承的层级不要超过2-3层。
  • 优先考虑组合(Composition)而非继承:这是面向对象设计中一个非常重要的原则——“优先使用组合,而不是继承”。当对象之间是“has-a”(拥有一个)的关系而不是严格的“is-a”(是一个)的关系时,组合通常是更好的选择。例如,一个
    Car
    “拥有”一个
    Engine
    ,而不是
    Car
    “是”一个
    Engine
    的子类。组合能提供更大的灵活性,减少类之间的耦合。
  • 单一职责原则(SRP):每个类都应该只有一个改变的理由。父类和子类都应该专注于各自的核心职责。如果一个类承担了过多的职责,它就可能变得臃肿且难以维护。
  • 谨慎使用
    super
    关键字
    :确保你清楚
    super
    在构造函数和方法中的不同行为。在子类构造函数中,必须先调用
    super()
    才能使用
    this
  • 考虑Mixins或高阶函数:对于需要共享行为但不是严格继承关系的场景,Mixins(通过函数组合或ES6类的方法注入)或高阶函数(HOCs)常常是更灵活、更解耦的替代方案。

继承是一个强大的工具,但像所有工具一样,它需要被正确地使用。在实践中,我们应该根据具体的需求和设计原则,权衡继承与组合的优劣,选择最适合的方案。

魔法映像企业网站管理系统
魔法映像企业网站管理系统

技术上面应用了三层结构,AJAX框架,URL重写等基础的开发。并用了动软的代码生成器及数据访问类,加进了一些自己用到的小功能,算是整理了一些自己的操作类。系统设计上面说不出用什么模式,大体设计是后台分两级分类,设置好一级之后,再设置二级并选择栏目类型,如内容,列表,上传文件,新窗口等。这样就可以生成无限多个二级分类,也就是网站栏目。对于扩展性来说,如果有新的需求可以直接加一个栏目类型并新加功能操作

下载

JS类继承与原型链继承的本质区别和联系是什么?

要深入理解JS的类继承,就不能绕开它与原型链继承的关系。我个人认为,它们的本质联系远大于区别,而所谓的“区别”更多体现在语法糖和开发者体验上。

本质联系:

  • class
    是原型链的“语法糖”
    :这是最核心的联系。ES6的
    class
    关键字并没有引入一套全新的继承机制。JavaScript的继承机制从诞生之日起就是基于原型的。
    class
    语法只是提供了一个更高级、更友好的抽象层,让开发者能够以更接近传统面向对象语言的方式来编写代码。在底层,当你使用
    extends
    关键字时,JavaScript引擎仍然在幕后操作原型链,将子类的
    [[Prototype]]
    (即
    __proto__
    )指向父类的原型对象。
    • 例如,当我们定义
      class Dog extends Animal {}
      时,实际上JS引擎会做类似
      Object.setPrototypeOf(Dog.prototype, Animal.prototype)
      这样的操作,确保
      Dog
      的实例能够通过原型链访问到
      Animal.prototype
      上的方法。
    • 同时,
      Dog
      本身(构造函数)也会继承
      Animal
      (构造函数)的静态属性和方法,这通过
      Object.setPrototypeOf(Dog, Animal)
      来实现。

主要区别(体现在语法、开发者体验和一些细节行为):

  1. 语法层面

    • 原型链继承(ES5及以前):通常需要手动操作

      prototype
      属性,例如:

      function Animal(name) {
        this.name = name;
      }
      Animal.prototype.eat = function() { console.log(`${this.name} eats.`); };
      
      function Dog(name, breed) {
        Animal.call(this, name); // 构造函数继承
        this.breed = breed;
      }
      // 原型链继承
      Dog.prototype = Object.create(Animal.prototype);
      Dog.prototype.constructor = Dog; // 修复constructor指向
      Dog.prototype.bark = function() { console.log(`${this.name} barks.`); };

      这种方式相对冗长,且需要注意

      constructor
      的修复。

    • 类继承(ES6

      class
      :使用简洁的
      class
      extends
      关键字,如前面示例所示。语法更清晰,一步到位。

      class Animal { /* ... */ }
      class Dog extends Animal { /* ... */ }
  2. super
    关键字

    • 类继承:引入了
      super
      关键字,使得调用父类构造函数和方法变得非常直接和安全。
      super()
      在子类构造函数中调用父类构造函数,
      super.methodName()
      调用父类方法,并且
      this
      的指向会被自动处理。
    • 原型链继承:没有
      super
      关键字。调用父类构造函数需要使用
      Parent.call(this, ...)
      ,调用父类方法需要通过
      Parent.prototype.methodName.call(this, ...)
      ,这需要开发者手动管理
      this
      的上下文,容易出错。
  3. 构造函数行为

    • 类继承:在子类构造函数中,你必须先调用
      super()
      才能使用
      this
      。这是因为
      super()
      负责初始化
      this
      ,它实际上是执行父类的构造函数并返回一个绑定了
      this
      的实例。
    • 原型链继承:在子类构造函数中,你可以选择在调用
      Parent.call(this, ...)
      之前或之后使用
      this
      ,但通常建议在调用之后,以确保父类属性已被初始化。
  4. 严格模式

    • class
      内部的代码默认运行在严格模式下,即使你没有显式声明
      'use strict'
      。这有助于避免一些常见的JavaScript陷阱,例如意外的全局变量创建。
    • 传统函数构造器则不默认处于严格模式。
  5. 不可枚举的方法

    • class
      中定义的方法(包括静态方法)默认是不可枚举的(
      enumerable: false
      )。
    • 而通过
      Function.prototype.method = function() {}
      添加到原型上的方法,默认是可枚举的(尽管通常我们会将它们设置为不可枚举)。

归根结底,它们都是JavaScript实现面向对象继承的方式。

class
语法可以看作是JavaScript语言发展过程中,为了提供更现代、更符合主流OOP习惯的编程范式而做出的语法层面的改进。它让开发者能够更专注于业务逻辑,而不用过多地纠结于原型链的底层细节。但作为JS开发者,了解其原型链的本质,仍然是深入理解这门语言的关键。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

831

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

737

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

80

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.5万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号