原型與原型鏈

一、什麼是原型

在JavaScript中,每個對象都有一個指向另一個對象的引用,叫做原型。原型是JavaScript中一個比較重要的概念。

通過使用構造函數創建的對象,會自動擁有一個原型對象。原型的作用就是用來繼承屬性和方法。

function Person(name, age) {
  this.name = name;
  this.age = age;
}

Person.prototype.sayHello = function() {
  console.log("Hello, " + this.name);
};

var person1 = new Person("Alice", 18);
var person2 = new Person("Bob", 20);

person1.sayHello(); // "Hello, Alice"
person2.sayHello(); // "Hello, Bob"

在這個例子中,我們定義了一個Person對象,通過Person的原型來增加了一個sayHello方法,然後我們根據這個構造函數來創建兩個不同的對象person1和person2,通過調用sayHello方法,我們可以看到兩個對象都可以正確的繼承到sayHello方法。

二、原型鏈的作用

原型是可以被繼承的,由此形成的繼承關係被稱為原型鏈。當對象調用方法或屬性時,如果本身找不到,就會去原型對象中尋找,如果還是找不到,則會繼續去原型對象的原型對象中查找,構成一個鏈式結構,直到最後查找完整個鏈才會返回undefined。

function Animal() {
  this.name = "Animal";
}

Animal.prototype.eat = function() {
  console.log(this.name + " is eating");
};

function Cat() {
  this.name = "Cat";
}

Cat.prototype = new Animal();
Cat.prototype.constructor = Cat;

var cat1 = new Cat();
cat1.eat(); // "Cat is eating"

在這個例子中,我們定義了一個Animal構造函數,然後在它的原型上添加一個eat方法。然後我們又定義了一個Cat構造函數,由於我們希望Cat繼承Animal的屬性和方法,所以我們將Cat的原型指向了Animal的實例,這樣一來Cat就可以調用到Animal的屬性和方法了。

當我們調用cat1的eat方法時,首先在cat1對象上尋找是否有eat屬性或方法,沒有找到,於是它會去cat1的原型對象Cat.prototype中查找,還是沒有找到,於是它又去Cat.prototype的原型對象Animal.prototype中查找,最終在Animal.prototype中找到了eat方法,然後執行。這就是原型鏈的查找過程。

三、原型鏈的細節

在JavaScript中,有些方法是自身屬性,有些是繼承屬性;也有一些屬性是自身的,有一些是繼承來的。由於原型鏈的繼承關係,有些情況下會出現一些意料之外的結果,下面舉幾個例子說明一下:

var str = "hello";
console.log(str.toString()); // "hello"

var arr = [1,2,3];
console.log(arr.toString()); // "1,2,3"

在這兩個例子中,我們在字符串和數組對象上調用了toString方法,然而這兩個對象並沒有自己的toString方法,是從它們的構造函數Object中繼承了toString方法。這也是為什麼其他對象也可以使用toString方法。

function Person(name, age) {
  this.name = name;
  this.age = age;
}

Person.prototype.sayHello = function() {
  console.log("Hello, " + this.name);
};

var person1 = new Person("Alice", 18);
var person2 = new Person("Bob", 20);

console.log("name" in person1); // true
console.log(person1.hasOwnProperty("name")); // true
console.log(person1.hasOwnProperty("sayHello")); // false 

console.log("sayHello" in person1); // true
console.log(person1.__proto__.hasOwnProperty("sayHello")); // true
console.log(person1.__proto__.hasOwnProperty("name")); // false

在這個例子中,我們通過person1對象來演示hasOwnProperty方法和in方法的區別。hasOwnProperty是檢測對象自身是否擁有某個屬性,而in方法是檢測對象是否擁有某個屬性,不管是自身還是繼承而來的。因為person1對象自己擁有name屬性,所以hasOwnProperty返回true,而因為sayHello方法是從它的原型對象Person.prototype上繼承而來的,所以hasOwnProperty返回false,in方法卻返回true。

四、如何組合使用原型與構造函數

在實際編程中,常常需要使用原型和構造函數來一起工作。比如,希望繼承某個對象,並在繼承的同時傳遞一些參數;或者希望封裝私有變量,但又能讓實例對象共享某些公共的屬性和方法等等。以下給出一些例子:

// 1、通過原型繼承屬性和方法,通過構造函數來傳遞參數
function Person(name, age) {
  this.name = name;
  this.age = age;
}

Person.prototype.sayHello = function() {
  console.log("Hello, " + this.name);
};

function Student(name, age, school) {
  Person.call(this, name, age);
  this.school = school;
}

Student.prototype = new Person();
Student.prototype.constructor = Student;

Student.prototype.sayHello = function() {
  console.log("Hello, I'm a student, my name is " + this.name);
};

var student1 = new Student("Alice", 18, "Harvard");
student1.sayHello(); // "Hello, I'm a student, my name is Alice"

// 2、使用原型來封裝私有變量
function Counter() {
  var count = 0;
  this.getCount = function() {
    return count;
  };
}

Counter.prototype.increment = function() {
  var count = this.getCount();
  count++;
  this.getCount = function() {
    return count;
  };
};

var counter1 = new Counter();
counter1.increment();
console.log(counter1.getCount()); // 1

var counter2 = new Counter();
counter2.increment();
console.log(counter2.getCount()); // 1,而不是2

以上這兩個例子都非常常見。第一個例子通過Student構造函數來傳遞參數,同時繼承了Person的屬性和方法;第二個例子封裝了私有變量,並通過原型來共享increment方法。

五、總結

原型和原型鏈是JavaScript中比較重要的概念,通過靈活運用原型和構造函數,我們可以很方便地實現繼承、私有變量等功能。同時,需要注意的是,由於原型鏈的存在,有些方法可能並不是對象自身的,而是繼承而來的,因此在編程時需要特別注意。

原創文章,作者:GNIGG,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/360768.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
GNIGG的頭像GNIGG
上一篇 2025-02-24 00:33
下一篇 2025-02-24 00:33

相關推薦

  • 谷歌Axure插件——提高原型設計效率的利器

    一、快速製作自定義組件 1、Axure自帶的組件庫雖然豐富,但在某些情況下我們需要自己設計一些特殊的組件,而谷歌Axure插件提供了一個輕鬆快捷的方式。 以設計一個自定義開關組件為…

    編程 2025-04-22
  • 反思原型鏈

    一、原型鏈基礎 JavaScript的原型鏈是一種從對象到另一個對象的委託關係。它在一個對象的屬性找不到時,會沿着原型鏈往上找指定名稱的屬性或方法。我們來看一個例子: functi…

    編程 2025-04-12
  • 深入解析JavaScript原型鏈面試題

    JavaScript 是一種弱類型、基於原型的編程語言。原型是 JavaScript 的一項非常重要的特性,也是面試中經常考察的知識點。JS 原型鏈是基於原型的面向對象編程的基石,…

    編程 2025-03-12
  • 原型模式的應用場景

    一、單例模式的應用場景 單例模式是一種常見的設計模式,其應用場景也非常廣泛。單例模式通常只允許一個實例存在,可以用於創建全局唯一的對象。在使用原型模式實現單例模式時,先創建一個原型…

    編程 2025-02-17
  • js原型鏈編程,js原型鏈概念

    本文目錄一覽: 1、javascript 原型,原型鏈是什麼?有什麼特點 2、js原型和原型鏈的理解是什麼? 3、js原型鏈和繼承的理解 4、談談對原型鏈的理解 JS原型鏈怎麼理解…

    編程 2025-01-09
  • java桌面應用程序原型,軟件開發 原型

    本文目錄一覽: 1、Java原型模式 2、Java適合開發桌面應用程序嗎? 3、JAVA 怎麼把程序做成 桌面應用程序 Java原型模式 不用例子,如果你的某個類在整個工程中只需要…

    編程 2024-12-15
  • 原型網絡:從概念到應用

    一、什麼是原型網絡 原型網絡(Prototypical Network)是一種深度學習中用於學習表示空間的神經網絡,由於能夠處理較小的數據集而被廣泛應用於圖像分類、目標識別等領域。…

    編程 2024-12-12
  • 軟件原型設計詳解

    一、軟件原型設計工具 軟件原型設計是實現產品設計的重要步驟,而軟件原型設計工具就是我們能夠使用的工具。 目前市面上存在很多軟件原型設計工具,比較常見的有: Axure RP Ske…

    編程 2024-11-25
  • c語言中c=a\u003e3,C語言中,函數可以用原型說明,也可用簡單說明

    本文目錄一覽: 1、c語言中 c=”a”是什麼意思 2、在c語言中char choose=0是什麼意思 3、C語言中 c=’a’, …

    編程 2024-11-08
  • c語言strcmp原型,c語言strcpy函數原型

    本文目錄一覽: 1、strcmp原理 2、C語言中strcmp函數怎麼用啊 3、C語言編程strcmp函數的問題 4、C語言 strcmp(“how”,&#…

    編程 2024-11-05

發表回復

登錄後才能評論