”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 提升你的 JavaScript:深入研究面向对象编程✨

提升你的 JavaScript:深入研究面向对象编程✨

发布于2024-11-07
浏览:472

Elevate Your JavaScript: A Deep Dive into Object-Oriented Programming✨

面向对象编程 (OOP) 是一种强大的范例,它彻底改变了我们构建和组织代码的方式。

虽然 JavaScript 最初是一种基于原型的语言,但它已经发展到接受 OOP 原则,特别是随着 ES6 的引入和后续更新。

这篇文章深入研究了 JavaScript 中 OOP 的核心概念,探索如何实现它们来创建更健壮、可维护和可扩展的应用程序。

我们将了解 OOP 的四大支柱 - 继承、抽象、封装和多态性 - 演示如何在 JavaScript 中应用每个原则。在此过程中,我们将研究现实世界的示例并讨论每个概念的优缺点。

无论您是希望提高 JavaScript 中 OOP 技能的经验丰富的开发人员,还是渴望掌握这些基本概念的新手,本指南都将为您在 JavaScript 项目中利用 OOP 的力量提供宝贵的见解。


1。遗产:

继承允许一个类从另一个类继承属性和方法。它提高了代码的可重用性并建立了父类和子类之间的关系。

class Vehicle {
  constructor(make, model) {
    this.make = make;
    this.model = model;
  }

  getInfo() {
    return `${this.make} ${this.model}`;
  }

  start() {
    return "The vehicle is starting...";
  }
}

class Car extends Vehicle {
  constructor(make, model, doors) {
    super(make, model);
    this.doors = doors;
  }

  getCarInfo() {
    return `${this.getInfo()} with ${this.doors} doors`;
  }
}

const myCar = new Car("Toyota", "Corolla", 4);
console.log(myCar.getCarInfo()); // Output: Toyota Corolla with 4 doors
console.log(myCar.start()); // Output: The vehicle is starting...

在此示例中,Car 继承自 Vehicle,获取对其属性和方法的访问权限。

优点:

  • 代码可重用性:子类继承父类的属性和方法。

  • 在对象之间建立清晰的层次结构。

  • 允许方法重写和扩展。

缺点:

  • 可能导致父类和子类之间的紧密耦合。

  • 深层继承层次结构可能会变得复杂且难以维护。


2.抽象

抽象涉及隐藏复杂的实现细节并仅显示对象的必要特征。在 JavaScript 中,我们可以使用抽象类(尽管本身不支持)和接口来实现抽象。

class Shape {
  constructor() {
    if (new.target === Shape) {
      throw new TypeError("Cannot instantiate abstract class");
    }
  }

  calculateArea() {
    throw new Error("Method 'calculateArea()' must be implemented.");
  }
}

class Circle extends Shape {
  constructor(radius) {
    super();
    this.radius = radius;
  }

  calculateArea() {
    return Math.PI * this.radius ** 2;
  }
}

class Rectangle extends Shape {
  constructor(width, height) {
    super();
    this.width = width;
    this.height = height;
  }

  calculateArea() {
    return this.width * this.height;
  }
}

// const shape = new Shape(); // Throws TypeError
const circle = new Circle(5);
const rectangle = new Rectangle(4, 6);

console.log(circle.calculateArea()); // Output: 78.53981633974483
console.log(rectangle.calculateArea()); // Output: 24

在此示例中,Shape 充当无法直接实例化的抽象类。它定义了所有子类都必须实现的公共接口calculateArea。这种抽象允许我们通过通用接口处理不同的形状,而不必担心它们的具体实现。

优点:

  • 通过隐藏不必要的细节来简化复杂的系统。

  • 提高代码可维护性并减少重复。

  • 允许关注对象做什么而不是它如何做。

缺点:

  • 如果设计不仔细,可能会导致过度简化。

  • 在某些情况下可能会带来性能开销。


3.封装

封装是将数据和在单个单元(对象)内操作该数据的方法捆绑在一起。在 JavaScript 中,我们可以使用闭包和符号来创建私有属性和方法。

class BankAccount {
  #balance = 0;  // Private field

  constructor(owner) {
    this.owner = owner;
  }

  deposit(amount) {
    if (amount > 0) {
      this.#balance  = amount;
      return true;
    }
    return false;
  }

  withdraw(amount) {
    if (amount > 0 && this.#balance >= amount) {
      this.#balance -= amount;
      return true;
    }
    return false;
  }

  getBalance() {
    return this.#balance;
  }
}

const account = new BankAccount('John Doe');
account.deposit(1000);
console.log(account.getBalance()); // Output: 1000
console.log(account.#balance); // SyntaxError: Private field '#balance' must be declared in an enclosing class

在此示例中,#balance 是私有字段,无法从类外部直接访问。

优点:

  • 数据保护:防止未经授权访问内部数据。

  • 模块化:将相关功能捆绑在一起。

  • 更容易维护:内部实现的更改不会影响外部代码。

缺点:

  • 由于缺乏真正的私有成员,在 JavaScript 中实现起来可能很复杂。

  • 创建 getter 和 setter 时可能会导致冗长的代码。


4。多态性

多态性允许将不同类的对象视为公共超类的对象。在 JavaScript 中,这可以通过方法重写来实现。

class Animal {
  speak() {
    return "The animal makes a sound";
  }
}

class Dog extends Animal {
  speak() {
    return "The dog barks";
  }
}

class Cat extends Animal {
  speak() {
    return "The cat meows";
  }
}

const animals = [new Animal(), new Dog(), new Cat()];

animals.forEach(animal => {
  console.log(animal.speak());
});

// Output:
// The animal makes a sound
// The dog barks
// The cat meows

在这个例子中,每个类以不同的方式实现speak方法,展示了多态性。

优点:

  • 灵活性:不同类型的对象可以统一处理。

  • 可扩展性:可以在不更改现有代码的情况下添加新类。

  • 通过允许对不同类型使用单一接口来简化代码。

缺点:

  • 如果过度使用,会使代码更难调试。

  • 在某些语言中可能会导致性能开销(在 JavaScript 中较少)。


正如我们所探索的,JavaScript 中的面向对象编程提供了一个强大的工具包,用于创建结构化、可维护和可扩展的代码。 OOP 的四大支柱 - 继承、抽象、封装和多态性 - 每一个都带来独特的优势,允许开发人员建模复杂的系统、保护数据完整性、促进代码重用以及创建灵活、可扩展的应用程序。

虽然由于语言的独特特性,在 JavaScript 中实现这些原则有时可能需要创造性的方法,但好处是显而易见的。 OOP 可以使代码库更有组织性、团队成员之间的协作更轻松,并提高对不断变化的需求的适应性。

然而,重要的是要记住 OOP 并不是一种万能的解决方案。每个项目可能需要对这些原则进行不同的平衡,并且在某些情况下,其他范例可能更合适。关键是要彻底理解这些概念并明智地应用它们,始终牢记您的项目和团队的具体需求。

快乐编码?

版本声明 本文转载于:https://dev.to/alaa-samy/elevate-your-javascript-a-deep-dive-into-object-oriented-programming-2080?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 从命令行运行时,如何解决 Maven 中的“NoClassDefFoundError”问题?
    从命令行运行时,如何解决 Maven 中的“NoClassDefFoundError”问题?
    通过 Shade 插件解决 NoClassDefFoundError 的依赖关系在第一次使用 Maven 时,Eclipse 和命令行使用之间可能会出现不一致,原因是依赖管理。尽管在 Eclipse 中成功创建了 Maven 项目并添加了依赖项,但通过命令行执行项目可能会导致 NoClassDefF...
    编程 发布于2024-11-08
  • 如何在 Mac 上的 XAMPP 中安装 PHP 的国际扩展?
    如何在 Mac 上的 XAMPP 中安装 PHP 的国际扩展?
    Mac OS 上 XAMPP 的 PHP-intl 安装要解决在 Mac 上使用 XAMPP 时在 php 中安装 intl 扩展的问题,请按照以下步骤操作:确认PHP路径:执行哪个php来确定使用的PHP路径。对于 XAMPP,它应该是 /Applications/XAMPP/xamppfiles...
    编程 发布于2024-11-08
  • 如何在 Python 中删除列表元素的尾随字符?
    如何在 Python 中删除列表元素的尾随字符?
    拆分​​列表元素在编程中,经常需要将列表元素拆分为多个组件。一种常见的情况涉及删除尾随字符。假设您有一个字符串列表,其中每个元素都包含一个制表符 ('\t'),后跟其他文本。目标是消除此选项卡及其后面的所有内容,仅保留选项卡之前的文本。考虑以下列表:my_list = ['eleme...
    编程 发布于2024-11-08
  • 以下是根据您的具体要求为您的文章提供的一些标题选项:

* 为什么我的代码不起作用?理解 C++ 中的函数作用域
* C++ 中的函数作用域:为什么我的 HelloWorld() F
    以下是根据您的具体要求为您的文章提供的一些标题选项: * 为什么我的代码不起作用?理解 C++ 中的函数作用域 * C++ 中的函数作用域:为什么我的 HelloWorld() F
    C 中函数声明的范围 在您的代码中,您收到编译错误,因为 HelloWorld() 函数未在与调用它的范围相同。让我们深入研究一下函数作用域的概念并解决这个问题。函数原型,也称为声明,在不提供函数定义的情况下告知编译器函数的存在。在给定的代码中,您尝试调用 HelloWorld(),而不首先在当前作...
    编程 发布于2024-11-08
  • 深入研究 Monty Hall 问题项目:模拟和理解概率概念
    深入研究 Monty Hall 问题项目:模拟和理解概率概念
    欢迎来到 Monty Hall 问题模拟项目的迷人世界!这种实践学习经验将指导您创建基于网络的模拟,该模拟基于流行的游戏节目场景演示有趣的概率谜题。 揭开蒙蒂霍尔问题之谜 蒙蒂·霍尔问题是一个著名的概率难题,几十年来一直让人们感到困惑和着迷。通过参与这个项目,您不仅有机会实现模拟,还...
    编程 发布于2024-11-08
  • 如何在 PHP 中验证 MySQL DELETE 查询是否成功?
    如何在 PHP 中验证 MySQL DELETE 查询是否成功?
    验证 MySQL DELETE 查询是否成功执行 DELETE 操作时,确定其成功执行至关重要。在 PHP 中,您可以采用各种方法来确定 DELETE 查询是否成功。MySQLi 和 PDO使用 MySQLi 或 PDO、mysql_query() 和 PDO::成功删除查询后,exec() 返回不...
    编程 发布于2024-11-08
  • 如何在 Node.js 中提前退出 forEach 循环?
    如何在 Node.js 中提前退出 forEach 循环?
    如何中断 Node.js forEach 循环在需要递归遍历嵌套数据结构并对每个元素执行操作的情况下,您可以使用递归和forEach 的组合。但是,在某些情况下,您可能需要提前退出 forEach 循环。与带有 break 或 continue 语句的常规循环不同,forEach 缺乏停止迭代的内置...
    编程 发布于2024-11-08
  • Day f Brylnt:Next.js 与 Remix
    Day f Brylnt:Next.js 与 Remix
    大家好!我知道这与 Brylnt 的制作并不直接相关,但在决定使用哪个框架时我遇到了一些问题,我想我应该分享一下我对两个流行竞争者的想法:Next.js 和 混音. 这两个框架都很优秀,根据项目的不同,任何一个都可能是正确的选择。由于我使用的是 T3 Stack,其中包括 Next.js,我自然倾向...
    编程 发布于2024-11-08
  • 学习 CSS 网格:包含大量示例的简单指南
    学习 CSS 网格:包含大量示例的简单指南
    Hey there! If you've ever felt like CSS Grid is a bit like trying to solve a Rubik's Cube blindfolded, you're not alone. I'm Eleftheria, and today, I'...
    编程 发布于2024-11-08
  • 如何在 JavaScript 中强制刷新网页并绕过缓存?
    如何在 JavaScript 中强制刷新网页并绕过缓存?
    使用 JavaScript 硬刷新当前页面强制 Web 浏览器通过 JavaScript 硬刷新页面可确保获取页面的全新副本并更新其所有外部资源。 为了实现这一点,JavaScript 提供了一个名为 location.reload(true) 的方法。当传递 true 值时,此方法指示浏览器绕过其...
    编程 发布于2024-11-08
  • 什么是 PATH_INFO 以及它如何增强 PHP Web 应用程序?
    什么是 PATH_INFO 以及它如何增强 PHP Web 应用程序?
    深入研究 PATH_INFO:揭示其在 PHP Web 应用程序中的作用在 Web 开发领域,PHP 是一个强大的工具,用于创建动态和交互式应用程序。它的全部功能之一是名为 PATH_INFO 的神秘变量。尽管经常被提及,但许多人仍然难以理解其确切作用。本文深入研究 PATH_INFO,阐明其目的、...
    编程 发布于2024-11-08
  • 如何使用 Connector .NET 检索 MySQL 中的最后一个插入 ID?
    如何使用 Connector .NET 检索 MySQL 中的最后一个插入 ID?
    使用 Connector .NET 在 MySql 中检索最后一个插入 ID在 MySql 中,最后一个插入 ID 是指分配给新插入的标识符排。该值在某些情况下可能很有价值,例如填充外键关系。最初,假设 MySqlHelper 类的 ExecuteNonQuery 方法返回最后一个插入 ID。然而,...
    编程 发布于2024-11-08
  • 如何在 PHP 中使用 cURL 获取 API 响应?
    如何在 PHP 中使用 cURL 获取 API 响应?
    在 PHP 中使用 cURL 获取 API 响应在 PHP 中,您可以创建一个独立的类,其中包含通过 cURL 调用 API 的函数并获得响应。以下是实现此目的的方法:class ApiRequest { public function getResponse($url) { // Se...
    编程 发布于2024-11-08
  • Ansible 入门 - 初学者指南:日复一日的 DevOps 工具系列
    Ansible 入门 - 初学者指南:日复一日的 DevOps 工具系列
    欢迎来到我们的“50 天 50 个 DevOps 工具”系列的第 30 天!今天,我们将探索 Ansible,它是 DevOps 工具包中最重要的工具之一。本博客将向您介绍 Ansible 的基础知识,分解其关键组件并向您展示如何从简单的示例开始。我们会让事情简单明了,使其成为初学者的完美起点。 ...
    编程 发布于2024-11-08
  • 什么是CPU寄存器
    什么是CPU寄存器
    什么是寄存器: 计算机寄存器是计算机中央处理单元 (CPU) 内的小型高速存储单元,用于临时保存数据和指令,以便在处理过程中快速访问。它们是直接影响 CPU 运算速度和效率的重要组件。 寄存器的访问速度比内存更快,因为它们位于 CPU 内部。这种接近性允许更快的数据检索和处理。 寄存器内存是计算机中...
    编程 发布于2024-11-08

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3