」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 提升你的 JavaScript:深入研究物件導向程式設計✨

提升你的 JavaScript:深入研究物件導向程式設計✨

發佈於2024-11-07
瀏覽:972

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]刪除
最新教學 更多>
  • Java為何無法創建泛型數組?
    Java為何無法創建泛型數組?
    通用陣列創建錯誤 arrayList [2]; JAVA報告了“通用數組創建”錯誤。為什麼不允許這樣做? 答案:Create an Auxiliary Class:public static ArrayList<myObject>[] a = new ArrayList<my...
    程式設計 發佈於2025-05-15
  • 如何將來自三個MySQL表的數據組合到新表中?
    如何將來自三個MySQL表的數據組合到新表中?
    mysql:從三個表和列的新表創建新表 答案:為了實現這一目標,您可以利用一個3-way Join。 選擇p。 *,d.content作為年齡 來自人為p的人 加入d.person_id = p.id上的d的詳細信息 加入T.Id = d.detail_id的分類法 其中t.taxonomy ...
    程式設計 發佈於2025-05-15
  • 如何將PANDAS DataFrame列轉換為DateTime格式並按日期過濾?
    如何將PANDAS DataFrame列轉換為DateTime格式並按日期過濾?
    Transform Pandas DataFrame Column to DateTime FormatScenario:Data within a Pandas DataFrame often exists in various formats, including strings.使用時間數據時...
    程式設計 發佈於2025-05-15
  • 如何使用組在MySQL中旋轉數據?
    如何使用組在MySQL中旋轉數據?
    在關係數據庫中使用mySQL組使用mySQL組進行查詢結果,在關係數據庫中使用MySQL組,轉移數據的數據是指重新排列的行和列的重排以增強數據可視化。在這裡,我們面對一個共同的挑戰:使用組的組將數據從基於行的基於列的轉換為基於列。 Let's consider the following ...
    程式設計 發佈於2025-05-15
  • 查找當前執行JavaScript的腳本元素方法
    查找當前執行JavaScript的腳本元素方法
    如何引用當前執行腳本的腳本元素在某些方案中理解問題在某些方案中,開發人員可能需要將其他腳本動態加載其他腳本。但是,如果Head Element尚未完全渲染,則使用document.getElementsbytagname('head')[0] .appendChild(v)的常規方...
    程式設計 發佈於2025-05-15
  • `console.log`顯示修改後對象值異常的原因
    `console.log`顯示修改後對象值異常的原因
    foo = [{id:1},{id:2},{id:3},{id:4},{id:id:5},],]; console.log('foo1',foo,foo.length); foo.splice(2,1); console.log('foo2', foo, foo....
    程式設計 發佈於2025-05-15
  • CSS強類型語言解析
    CSS強類型語言解析
    您可以通过其强度或弱输入的方式对编程语言进行分类的方式之一。在这里,“键入”意味着是否在编译时已知变量。一个例子是一个场景,将整数(1)添加到包含整数(“ 1”)的字符串: result = 1 "1";包含整数的字符串可能是由带有许多运动部件的复杂逻辑套件无意间生成的。它也可以是故意从单个真理...
    程式設計 發佈於2025-05-15
  • 如何使用Depimal.parse()中的指數表示法中的數字?
    如何使用Depimal.parse()中的指數表示法中的數字?
    在嘗試使用Decimal.parse(“ 1.2345e-02”中的指數符號表示法表示的字符串時,您可能會遇到錯誤。這是因為默認解析方法無法識別指數符號。 成功解析這樣的字符串,您需要明確指定它代表浮點數。您可以使用numbersTyles.Float樣式進行此操作,如下所示:[&& && && ...
    程式設計 發佈於2025-05-15
  • 如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    如何為PostgreSQL中的每個唯一標識符有效地檢索最後一行?
    postgresql:為每個唯一標識符在postgresql中提取最後一行,您可能需要遇到與數據集合中每個不同標識的信息相關的信息。考慮以下數據:[ 1 2014-02-01 kjkj 在數據集中的每個唯一ID中檢索最後一行的信息,您可以在操作員上使用Postgres的有效效率: id dat...
    程式設計 發佈於2025-05-15
  • 如何避免Go語言切片時的內存洩漏?
    如何避免Go語言切片時的內存洩漏?
    ,a [j:] ...雖然通常有效,但如果使用指針,可能會導致內存洩漏。這是因為原始的備份陣列保持完整,這意味著新切片外部指針引用的任何對象仍然可能佔據內存。 copy(a [i:] 對於k,n:= len(a)-j i,len(a); k
    程式設計 發佈於2025-05-15
  • 如何干淨地刪除匿名JavaScript事件處理程序?
    如何干淨地刪除匿名JavaScript事件處理程序?
    刪除匿名事件偵聽器將匿名事件偵聽器添加到元素中會提供靈活性和簡單性,但是當要刪除它們時,可以構成挑戰,而無需替換元素本身就可以替換一個問題。 element? element.addeventlistener(event,function(){/在這里工作/},false); 要解決此問題,請考...
    程式設計 發佈於2025-05-15
  • 在GO中構造SQL查詢時,如何安全地加入文本和值?
    在GO中構造SQL查詢時,如何安全地加入文本和值?
    在go中構造文本sql查詢時,在go sql queries 中,在使用conting and contement和contement consem per時,尤其是在使用integer per當per當per時,per per per當per. [&​​&&&&&&&&&&&&&&&默元組方法在...
    程式設計 發佈於2025-05-15
  • Java中Lambda表達式為何需要“final”或“有效final”變量?
    Java中Lambda表達式為何需要“final”或“有效final”變量?
    Lambda Expressions Require "Final" or "Effectively Final" VariablesThe error message "Variable used in lambda expression shou...
    程式設計 發佈於2025-05-15
  • 如何有效地選擇熊貓數據框中的列?
    如何有效地選擇熊貓數據框中的列?
    在處理數據操作任務時,在Pandas DataFrames 中選擇列時,選擇特定列的必要條件是必要的。在Pandas中,選擇列的各種選項。 選項1:使用列名 如果已知列索引,請使用ILOC函數選擇它們。請注意,python索引基於零。 df1 = df.iloc [:,0:2]#使用索引0和1 ...
    程式設計 發佈於2025-05-15
  • 如何在Java字符串中有效替換多個子字符串?
    如何在Java字符串中有效替換多個子字符串?
    在java 中有效地替換多個substring,需要在需要替換一個字符串中的多個substring的情況下,很容易求助於重複應用字符串的刺激力量。 However, this can be inefficient for large strings or when working with nu...
    程式設計 發佈於2025-05-15

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3