目錄
 物件字面量
 工廠函數
 原型鏈
 ES6 類別
 比較
  效能
  特點
 結論
首頁 web前端 js教程 JavaScript 建立物件: 方法一覽與最佳實踐

JavaScript 建立物件: 方法一覽與最佳實踐

Feb 23, 2017 pm 01:30 PM



  在JavaScript中「建立物件」是一個複雜的話題。這門語言提供了許多創造物件的方式,不論新手或老手都可能對此感到無所適從,不知道該選擇哪一種。不過,儘管創建物件的方法很多,看起來語法差異也很大,但實際上它們的相似性可能比你所以為的要多。本文將帶領你踏上一段梳理物件創建方法的旅程,為你揭示不同方法之間的依賴與遞進關係。

 物件字面量

  我們的第一站毫無疑問就是創建物件最簡單的方法,物件字面量。 JavaScript總是宣揚自己能夠「無中生有」地創造物件——不需要類別、不需要模板、不需要原型——「噌」地一下,一個有方法有資料的物件就出現了。

var o = {
  x: 42,
  y: 3.14,
  f: function() {},
  g: function() {}
};
登入後複製

  但這種方法有一個缺點:如果我們想在其他地方創建一個同類型的對象,就得把這個對象的方法、資料和初始化都複製貼上過去。我們需要一種能夠批量創建同類型物件的方法,而不是只創建一個物件。

 工廠函數

  我們的下一站是工廠函數。顯然,用這種方法來建立一類具有相同結構、介面和實作的物件是最簡單的。我們不直接建立一個物件字面量,而是將物件字面量作為函數的回傳值,當我們需要多次或多處建立同類型的物件時,只要呼叫這個函數就行了。

function thing() {
  return {
    x: 42,
    y: 3.14,
    f: function() {},
    g: function() {}
  };
}

var o = thing();
登入後複製

  但這種方法也有一個缺點:它會導致記憶體膨脹,因為每個物件都包含了工廠函數的獨立副本。理論上我們希望所有物件共享一個工廠函數副本。

 原型鏈

  JavaScript提供了一個內建的在物件之間共享資料的機制,稱為原型鏈。當我們存取一個物件的屬性時,它會委託某些其他物件來完成這項請求。我們可以利用這一點來修改工廠函數,使它創建的每個物件只包含自己特有的數據,而對其他屬性的請求則全部委託給原型鏈上共有的一個物件。

var thingPrototype = {
  f: function() {},
  g: function() {}
};

function thing() {
  var o = Object.create(thingPrototype);

  o.x = 42;
  o.y = 3.14;

  return o;
}

var o = thing();
登入後複製

  事實上,JavaScript本身就有內建的機制來支援這種通用模式。我們不需要自己創建這個共有的對象(即原型對象),JavaScript會自動為每個函數創建一個原型對象,我們可以把共享資料直接放在這個對象裡。

thing.prototype.f = function() {};
thing.prototype.g = function() {};

function thing() {
  var o = Object.create(thing.prototype);

  o.x = 42;
  o.y = 3.14;

  return o;
}

var o = thing();
登入後複製

  但這種方法也有一個缺點:會導致重複。上述thing函數的第一行和最後一行在每一個「委託原型的工廠函數」中都會重複一次,幾乎沒有差別。

 ES5 類別

  我們可以把那些重複的程式碼抽出來,放進一個自訂函數裡。這個函數會建立一個對象,並與其他某個任意函數(參數函數)的原型建立委託(繼承)關係,然後我們把新建立的物件當作參數,呼叫這個函數(參數函數),最後返回這個新的對象。

function create(fn) {
  var o = Object.create(fn.prototype);

  fn.call(o);

  return o;
}

// ...

Thing.prototype.f = function() {};
Thing.prototype.g = function() {};

function Thing() {
  this.x = 42;
  this.y = 3.14;
}

var o = create(Thing);
登入後複製

  事實上,JavaScript對此方法也有內建的支援機制。我們定義的這個create函數其實就是new關鍵字的一個基本實現,因此我們可以順手把create換成new。

Thing.prototype.f = function() {};
Thing.prototype.g = function() {};

function Thing() {
  this.x = 42;
  this.y = 3.14;
}

var o = new Thing();
登入後複製

  我們現在抵達的這一站通常被稱為ES5類。它透過函數來創建對象,把需要共享的資料委託給原型對象,並使用new關鍵字來處理重複的邏輯。

  但這種方法也有一個缺點:冗長又難看,而且在實現繼承的時候會更冗長更難看。

 ES6 類別

  JavaScript最新的相關改進是ES6 類,用新語法來實現上述功能要簡潔得多。

class Thing {
  constructor() {
    this.x = 42;
    this.y = 3.14;
  }

  f() {}
  g() {}
}

var o = new Thing();
登入後複製

 比較

  多年來,JavaScript開發者們與原型鏈的關係總是若即若離,糾纏不清。而今天我們最有可能遇到的兩種創建物件的方式,一種是強烈依賴原型鏈的class語法,另一種則是完全不依賴原型鏈的工廠函數語法。這兩種方式在性能上和特點上是不一樣的——儘管差異不太大。

  效能

  今天的JavaScript引擎已經經過了大幅度的優化,以至於很難透過JavaScript程式碼來推斷怎樣會比較快。關鍵在於測量方法。然而測量方法有時也會失靈。通常每六週就會有更新的JavaScript引擎發布,而在這之前採取的測量方法,和基於這種測量方法所做的決策都有可能失去意義。因此,我的經驗法則是選擇最官方、最廣泛使用的語法,因為大多數時候它經歷的實踐檢驗最多,因而性能是最高的。目前來說class語法最符合這一點,寫這篇文章時,class語法大約比回傳字面量的工廠函數快3倍。

  特點

  隨著ES6的發布,類別與工廠函數之間曾經存在的幾點差異消失了。現在,工廠函數和類別都能夠強制實現真正的私有資料-工廠函數透過閉包實現,類別透過WeakMap實現。兩者都能實現多重繼承——工廠函數可以將其他屬性混入自己的對象,類別也可以將其他屬性混入自己的原型,或者透過類別工廠,透過代理也能實現。工廠函數和類別也都可以在需要的時候傳回任意對象,語法也都很簡單。

 結論

  綜合考慮,我比較傾向於class文法。它標準、簡單、乾淨、快速,也提供了所有曾經只有函數工廠才具備的特質。

 以上就是JavaScript 創建物件: 方法一覽與最佳實踐的內容,更多相關內容請關注PHP中文網(www.php.cn)! 


#
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

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

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

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

前端熱敏紙小票打印遇到亂碼問題怎麼辦? 前端熱敏紙小票打印遇到亂碼問題怎麼辦? Apr 04, 2025 pm 02:42 PM

前端熱敏紙小票打印的常見問題與解決方案在前端開發中,小票打印是一個常見的需求。然而,很多開發者在實...

神秘的JavaScript:它的作用以及為什麼重要 神秘的JavaScript:它的作用以及為什麼重要 Apr 09, 2025 am 12:07 AM

JavaScript是現代Web開發的基石,它的主要功能包括事件驅動編程、動態內容生成和異步編程。 1)事件驅動編程允許網頁根據用戶操作動態變化。 2)動態內容生成使得頁面內容可以根據條件調整。 3)異步編程確保用戶界面不被阻塞。 JavaScript廣泛應用於網頁交互、單頁面應用和服務器端開發,極大地提升了用戶體驗和跨平台開發的靈活性。

誰得到更多的Python或JavaScript? 誰得到更多的Python或JavaScript? Apr 04, 2025 am 12:09 AM

Python和JavaScript開發者的薪資沒有絕對的高低,具體取決於技能和行業需求。 1.Python在數據科學和機器學習領域可能薪資更高。 2.JavaScript在前端和全棧開發中需求大,薪資也可觀。 3.影響因素包括經驗、地理位置、公司規模和特定技能。

JavaScript難以學習嗎? JavaScript難以學習嗎? Apr 03, 2025 am 12:20 AM

學習JavaScript不難,但有挑戰。 1)理解基礎概念如變量、數據類型、函數等。 2)掌握異步編程,通過事件循環實現。 3)使用DOM操作和Promise處理異步請求。 4)避免常見錯誤,使用調試技巧。 5)優化性能,遵循最佳實踐。

如何實現視差滾動和元素動畫效果,像資生堂官網那樣?
或者:
怎樣才能像資生堂官網一樣,實現頁面滾動伴隨的動畫效果? 如何實現視差滾動和元素動畫效果,像資生堂官網那樣? 或者: 怎樣才能像資生堂官網一樣,實現頁面滾動伴隨的動畫效果? Apr 04, 2025 pm 05:36 PM

實現視差滾動和元素動畫效果的探討本文將探討如何實現類似資生堂官網(https://www.shiseido.co.jp/sb/wonderland/)中�...

如何使用JavaScript將具有相同ID的數組元素合併到一個對像中? 如何使用JavaScript將具有相同ID的數組元素合併到一個對像中? Apr 04, 2025 pm 05:09 PM

如何在JavaScript中將具有相同ID的數組元素合併到一個對像中?在處理數據時,我們常常會遇到需要將具有相同ID�...

JavaScript的演變:當前的趨勢和未來前景 JavaScript的演變:當前的趨勢和未來前景 Apr 10, 2025 am 09:33 AM

JavaScript的最新趨勢包括TypeScript的崛起、現代框架和庫的流行以及WebAssembly的應用。未來前景涵蓋更強大的類型系統、服務器端JavaScript的發展、人工智能和機器學習的擴展以及物聯網和邊緣計算的潛力。

console.log輸出結果差異:兩次調用為何不同? console.log輸出結果差異:兩次調用為何不同? Apr 04, 2025 pm 05:12 PM

深入探討console.log輸出差異的根源本文將分析一段代碼中console.log函數輸出結果的差異,並解釋其背後的原因。 �...

See all articles