JavaScript原型實作繼承的方法介紹
在这篇文章中,我们将讨论原型以及如何在 JS 中使用它们进行继承。我们还将会看到原型方法与基于类的继承有何不同。
继承
继承是编程语言的一个显著特征,随着面向对象编程语言的引入而出现。这些语言大多是基于类的语言。在这里,类就像一个蓝图,对象是它的展现形式。就是说,要创建一个对象,首先我们必须创建一个类,然后我们可以从一个类创建任意数量的对象。
想象一下,我们有一个表示智能手机的类。这个类具有像其他智能手机一样的可以拍照、有GPS定位等功能。下面是使用 c 来描述这样的一个类:
class SmartPhone { public: void captureImages() {} } SmartPhone x; x.captureImages()
我们创建了一个名为SmartPhone
的类,它有一个名为capturePictures
的方法用来拍照。
如果我们需要一个iPhone
类,它可以捕捉图像和一些特殊的功能,比如面部ID扫描。下面是两种可能的解决方案:
将捕获图像功能与其他常见的智能手机功能,以及iPhone的特定功能一起重写到一个新类中。但是这种方法需要更多的时间和精力,并且会引入更多的bug。
- 重用
SmartPhone
类中的功能,这就是继承的作用,继承也是重用其他类/对象中功能的一种方式。
这里是我们如何从SmartPhone
类中继承capturePictures
方法,使用 c 实现如下:
class Iphone: public SmartPhone { public: void faceIDScan() {} } Iphone x x.faceIDScan() x.captureImages()
上面是一个简单的继承示例。 但是,它表明继承可以使我们以某种方式重用代码,从而使所生成的程序更不易出错,并且花费更少的时间进行开发。
以下是关于类的一些重要信息:
- 继承该功能的类称为子类
- 被继承的类称为父类
- 一个类可以同时从多个类中继承
- 我们可以具有多个继承级别。 例如,类C继承自类B,而类B继承自类A
值得注意的是,类本身并没有做任何事情。在从类创建对象之前,实际上没有完成任何工作。我们将看到它为什么不同于JavaScript。
原型是什么?
在 JS 中,所有对象都有一个特殊的内部属性,该属性基本上是对另一个对象的引用。 此引用取决于对象的创建方式。 在 ECMAScript/JavaScript规范中,它表示为[[Prototype]]
。
由于[[Prototype]]
链接到一个对象,所以该对象有自己的[[Prototype]]
引用。这就是建立原型链的方式。
这个[[Prototype]]
链是 JS 中继承的构建块。
__proto__
对象
为了访问对象的[[Prototype]]
,大多数浏览器都提供__proto__
属性。访问方式如下:
obj.__proto__
需要注意的是,这个属性不是 ECMAScript 标准的一部分,它实际上是由浏览器实现的。
获取和设置原型方法
除了__proto__
属性外,还有一种访问[[Prototype]]
的标准方法:
Object.getPrototypeOf(obj);
对应的有个类似的方法来设置对象的[[Prototype]]
:
Object.setPrototypeOf(obj, prototype);
[[Prototype]]
和.prototype
属性
[[Prototype]]
只不过是一种用来表示物体原型的标准符号。 许多开发人员将其与.prototype
属性混淆,这是完全不同的事情,接着我们来研究一下.prototype
属性。
在 JS 中,有许多创建对象的方法。一种方法是使用构造函数,像这样使用new
关键字来调用它:
function SmartPhone(os) { this.os = os } let phone = new SmartPhone('Android')
在控制台打印 phone
对象:
{ os: "IPhone" __proto__{ constructor: ƒ SmartPhone(os) __proto__: Object } }
现在,如果我们希望在phone
对象上有一些方法,我们可以在函数上使用.prototype
属性,如下所示:
SmartPhone.prototype.isAndroid = function () { return this.os === 'Android' || 'android' }
再次创建phone
对象时,打印 phone
对象如下:
{ os: "Android" __proto__{ isAndroid: ƒ() constructor: ƒ SmartPhone(os) __proto__: Object } }
我们可以在对象的[[Prototype]]
中看到isAndroid()
方法。
简而言之,.prototype
属性基本上就像由给定的构造函数创建的[[Prototype]]
对象的蓝图。 在.prototype
属性/对象中声明的所有内容都会在对象的[[Prototype]]
中弹出。
实上,如果将 SmartPhone.prototype 与
phone 的[[Prototype]]
进行比较,就会发现它们是相同的:
console.log(Object.getPrototypeOf(phone) === SmartPhone.prototype); // true
值得注意的是,我们还可以在构造函数中创建方法:
function ObjectA() { this.methodA = function () {} } let firstObj = new ObjectA() console.log(firstObj)
这种方法的问题是当我们初始化一个新对象时。所有实例都有自己methodA
的副本。相反,当我们在函数的原型上创建它时,对象的所有实例只共享方法的一个副本,显然使用原型的方式效率会过高。
当我们访问属性时这里发生了什么?
当我们访问一个属性以获取它时,会发生以下情况:
JS 引擎查找对象上的属性,如果找到了该属性,然后返回它。否则,JS 引擎将通过查看[[Prototype]]
来检查对象的继承属性,如果找到该属性,则返回它,否则,它会查找 [[Prototype]]
的[[Prototype]]
。 找到属性或没有[[Prototype]]
时,该链结束,这意味着我们已经到达原型链的末端。
当我们设置/创建属性时,JS 总是在对象本身上进行设置。 即使[[Prototype]]
链上存在相同的属性,下面是一个例子:
function MyObject() {} MyObject.prototype.propA = 10; // 在原型上创建属性 let myObject = new MyObject(); console.log(myObject.propA); // [[Prototype]]上的属性 // 10 myObject.propA = 20; // 对象的属性 console.log(myObject.propA); // 20
在上面的示例中,我们创建了一个构造函数,该函数的[[Prototype]]
上具有属性propA
。 当我们尝试对其进行读取操作时,会在控制台中看到该值。 但是,当我们尝试在对象本身上设置相同的属性时;JS 使用给定值在对象上创建一个新属性。 现在,如果我们不能直接访问[[Prototype]]
上的属性。
值得注意的是,普通对象的[[Prototype]]
链的末尾是内置的Object.prototype
。 这就是为什么大多数对象共享许多方法(例如toString()
)的原因。 因为它们实际上是在Object.prototype
上定义的。
使用原型继承的各种方法
在 JS 中,无论我们如何创建对象,只有原型继承,但这些方式还有一些区别,来看看:
对象字面量
在JavaScript中创建对象的最简单方法是使用对象字面量:
let obj = {}
如果在浏览器的控制台中打印obj
,我们将看到以下内容:
基本上,所有用文字面量创建的对象都继承了Object.prototype
的属性。
需要注意的是__proto__
对象引用了创建它的构造函数。 在这种情况下,constructor
属性指向Object
构造函数。
使用对象构造函数
另一种不太常见的创建对象的方法是使用对象构造函数。JS 提供了一个名为Object
的内置构造函数方法来创建对象。
let obj = new Object();
这种方法的结果与对象字面量的方式相同。它从Object.prototype
继承属性。因为我们使用Object
作为构造函数。
Object.create 方法
使用此辅助方法,我们可以创建一个带有[[Prototype]]
的对象,如下所示:
let SmartPhone = { captureImages: function() {} } let Iphone = Object.create(SmartPhone) Iphone.captureImages()
这是在 JS 中使用继承的最简单方法之一。猜猜我们如何在没有任何[[Prototype]]
引用的情况下创建对象?
构造方法
与 JS 运行时提供的对象构造函数相似。 我们还可以创建自己的构造函数,以创建适合我们需求的对象,如下所示:
function SmartPhone(os) { this.os = os; } SmartPhone.prototype.isAndroid = function() { return this.os === 'Android'; }; SmartPhone.prototype.isIOS = function() { return this.os === 'iOS'; };
现在,我们想创建一个iPhone
类,它应该有'iOS'
作为它 os
属性的值。它还应该有faceIDScan
方法。
首先,我们必须创建一个Iphone
构造函数,在其中,我们应该调用SmartPhone
构造函数,如下所示:
function Iphone() { SmartPhone.call(this, 'iOS'); }
这会将Iphone
构造函数中的this.os
属性设置为’iOS‘
。
之所以调用SmartPhone.call
方法,是因为我们需要更改 this
值以引用Iphone
。 这类似于在面向对象的世界中调用父级的构造函数。
接下来的事情是,我们必须从SmartPhone
构造函数继承方法。 我们可以在此处使用Object.create
朋友,如下所示:
Iphone.prototype = Object.create(SmartPhone.prototype);
现在,我们可以使用.prototype
为Iphone
添加方法,如下所示:
Iphone.prototype.faceIDScan = function() {};
最后,我们可以使用Iphone
创建一个对象,如下所示:
let x = new Iphone(); // calling inherited method console.log(x.isIOS()): // true
ES6 class
使用ES6,整个过程非常简单。 我们可以创建类(它们与C ++或其他任何基于类的语言中的类不同,只是在原型继承之上的语法糖),然后从其他类派生新的类。
下面是我们如何在ES6中创建类:
class SmartPhone { constructor(os) { this.os = os; } isAndroid() { return this.os === 'Android'; } isIos() { return this.os === 'iOS'; } };
现在,我们可以创建一个派生自SmartPhone
的新类,如下所示:
class Iphone extends SmartPhone { constructor() { super.call('iOS'); } faceIDScan() {} }
我们不是调用SmartPhone.call
,而是调用super.call
。 在内部,JavaScript引擎会自动为我们执行此操作。
最后,我们可以使用Iphone
创建一个对象,如下所示
let x = new Iphone(); x.faceIDScan(); // calling inherited method console.log(x.isIos()): // true
该ES6示例与先前的构造方法示例相同。 但是阅读和理解起来要干净得多。
原文:https://javascript.info/prototype-inheritance
作者:Indermohan Sing
更多编程相关知识,请访问:编程课程!!
以上是JavaScript原型實作繼承的方法介紹的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

在函數繼承中,使用「基底類別指標」和「衍生類別指標」來理解繼承機制:基底類別指標指向派生類別物件時,執行向上轉型,只存取基底類別成員。派生類別指標指向基底類別物件時,執行向下轉型(不安全),必須謹慎使用。

繼承錯誤調試技巧:確保正確的繼承關係。使用偵錯器逐步執行程式碼,檢查變數值。確保正確使用virtual修飾符。檢查隱藏的繼承帶來的菱形繼承問題。檢查抽象類別中未實現的純虛函數。

C++函式繼承詳解:掌握「is-a」和「has-a」關係什麼是函式繼承?函數繼承是C++中一種將衍生類別中定義的方法與基底類別中定義的方法關聯起來的技術。它允許衍生類別存取和重寫基底類別的方法,從而擴展了基底類別的功能。 「is-a」和「has-a」關係在函數繼承中,「is-a」關係指派生類別是基底類別的子類型,也就是說,衍生類別「繼承」了基底類別的特性和行為。 「has-a」關係指派生類別包含對基底類別物件的參考或指針,也就是說,衍生類別「擁有」了基底類別物件。語法以下是如何實作函數繼承的語法:classDerivedClass:pu

繼承和多態性會影響類別的耦合度:繼承會增加耦合度,因為衍生類別依賴基底類別。多態性可以降低耦合度,因為物件可以透過虛擬函數和基底類別指標以一致的方式回應訊息。最佳實踐包括謹慎使用繼承、定義公共介面、避免在基底類別中新增資料成員,以及透過依賴注入解耦類別。實戰案例顯示如何使用多態性和依賴注入來降低銀行帳戶應用程式中的耦合度。

原神4.4版新地圖介紹,小夥伴們原神這次4.4版本也是迎來了璃月的海燈節,同時將在4.4版本推出一個新的地圖區域,名為沉玉谷。根據提供的信息,沉玉谷實際上是翹英莊的一部分,但玩家更習慣稱其為沉玉谷。下面就讓小編來跟大家介紹一下新地圖吧。原神4.4版新地圖介紹4.4版本將開放璃月北部的「沉玉谷·上谷」、「沉玉谷·南陵」和「來歆山」,在「沉玉谷·上谷」已為旅行者開啟傳送錨點。 ※完成魔神任務序章·第三幕龍與自由之歌」後,將自動解鎖該傳送錨點。二、翹英莊當春日溫煦的柔風再度撫過沉玉的山野,那馥鬱的

什麼是物件導向程式設計?物件導向程式設計(OOP)是一種程式設計範式,它將現實世界中的實體抽象化為類,並使用物件來表示這些實體。類別定義了物件的屬性和行為,而物件則實例化了類別。 OOP的主要優點在於它可以使程式碼更易於理解、維護和重複使用。 OOP的基本概念OOP的主要概念包括類別、物件、屬性和方法。類別是物件的藍圖,它定義了物件的屬性和行為。物件是類別的實例,它具有類別的所有屬性和行為。屬性是物件的特徵,它可以儲存資料。方法是物件的函數,它可以對物件的資料進行操作。 OOP的優點OOP的主要優點包括:可重複使用性:OOP可以讓程式碼更

介面:無實作的契約介面在Java中定義了一組方法簽名,但不提供任何具體實作。它充當一種契約,強制實作該介面的類別實現其指定的方法。介面中的方法是抽象方法,沒有方法體。程式碼範例:publicinterfaceAnimal{voideat();voidsleep();}抽象類別:部分實作的藍圖抽象類別是一種父類,它提供了一個部分實現,可以被它的子類別繼承。與介面不同,抽象類別可以包含具體的實作和抽象方法。抽象方法是用abstract關鍵字聲明的,並且必須被子類別覆蓋。程式碼範例:publicabstractcla

函數繼承使衍生類別從基底類別繼承方法,可實現程式碼重用和多態性。遵循單一職責、開放-封閉、里斯替換原則設計良好繼承層次結構,可避免代碼耦合和鑽石問題。
