自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

為什么 JS 中的對(duì)象字面量很酷

開(kāi)發(fā) 前端
JS 中很多令人沮喪的問(wèn)題都是逐步解決的。本文演示了 ES 6 如何解決上述問(wèn)題,并使用額外的功能改進(jìn)對(duì)象字面量。

在 ES6 之前,JS 中的對(duì)象字面量(也稱(chēng)為對(duì)象初始化器)是非?;A(chǔ)的??梢远x兩種類(lèi)型的屬性:

  • 鍵值對(duì) {name1: value1}
  • 獲取器 { get name(){..} } 和 設(shè)置器 { set name(val){..}} 的計(jì)算屬性值

為什么 JS 中的對(duì)象字面量很酷

JS 是一種基于原型的語(yǔ)言,因此一切都是對(duì)象。在對(duì)象創(chuàng)建,配置和訪問(wèn)原型時(shí),必須提供一種易于構(gòu)造的語(yǔ)言。

定義一個(gè)對(duì)象并設(shè)置它的原型是一個(gè)常見(jiàn)的任務(wù)。最好的方式是直接在對(duì)象字面量使用一條語(yǔ)句來(lái)設(shè)置原型。

不幸的是,字面量的局限性不允許用一個(gè)簡(jiǎn)單的解決方案來(lái)實(shí)現(xiàn)這一點(diǎn)。必須結(jié)合使用object.create() 和對(duì)象字面量來(lái)設(shè)置原型。

為什么 JS 中的對(duì)象字面量很酷

我認(rèn)為這種解決方案不夠靈活。JS 是基于原型的,為什么要用原型創(chuàng)建對(duì)象那么麻煩?

幸運(yùn)的是,JS 也在慢慢完善。JS 中很多令人沮喪的問(wèn)題都是逐步解決的。

本文演示了 ES 6 如何解決上述問(wèn)題,并使用額外的功能改進(jìn)對(duì)象字面量。

  • 在對(duì)象構(gòu)造上設(shè)置原型
  • 方法的聲明
  • super 調(diào)用
  • 計(jì)算屬性名 

JS

1. 在對(duì)象構(gòu)造上設(shè)置原型

如你所知,訪問(wèn)現(xiàn)有對(duì)象原型的一種方法是使用 getter 屬性 __proto__:

  1. var myObject = { 
  2.   name: 'Hello World!', 
  3. }; 
  4. myObject.__proto__; // => {} 
  5. myObject.__proto__.isPrototypeOf(myObject); // => true 

myObject.__ proto__ 返回 myObject 的原型對(duì)象。

請(qǐng)注意,不建議將 object.__ proto__ 用作 getter/setter。替代方法應(yīng)考慮使用Object.getPrototypeOf() 和 Object.setPrototypeOf()。

ES6允許使用__proto__作為屬性名,并在 {__proto__:protoObject}中設(shè)置原型。

接著,咱們使用 __proto__ 屬性進(jìn)行對(duì)象初始化,并優(yōu)化上面的代碼:

  1. var myProto = { 
  2.   propertyExists: function(name) { 
  3.     return name in this; 
  4.   }, 
  5. }; 
  6. var myNumbers = { 
  7.   __proto__: myProto, 
  8.   array: [1, 6, 7], 
  9. }; 
  10. myNumbers.propertyExists('array'); // => true 
  11. myNumbers.propertyExists('collection'); // => false 

myNumbers 對(duì)象是使用特殊屬性名 proto 與創(chuàng)建原型 myProto,這次咱們使用一條語(yǔ)句就創(chuàng)建,沒(méi)有像上面還需要 object.create() 這樣的附加函數(shù)。

如你所看,使用 __proto__ 進(jìn)行編碼很簡(jiǎn)單,我一直喜歡簡(jiǎn)單明了的解決方案。

說(shuō)點(diǎn)脫離主題。我覺(jué)得奇怪的是,簡(jiǎn)單靈活的解決方案需要大量的工作和設(shè)計(jì)。如果解決方案很簡(jiǎn)單,你可能會(huì)認(rèn)為設(shè)計(jì)起來(lái)很容易。但是反之亦然:

  • 要使它簡(jiǎn)單明了是很復(fù)雜的
  • 把它變得復(fù)雜和難以理解是很容易的

如果某些東西看起來(lái)太復(fù)雜或難以使用,則可能還需要進(jìn)一步的完善。

你對(duì)簡(jiǎn)單性有何看法?(請(qǐng)?jiān)谙旅骐S意寫(xiě)評(píng)論)

2. proto 用法的特殊情況

即使__proto__看起來(lái)很簡(jiǎn)單,您也應(yīng)該注意一些特殊情況。

為什么 JS 中的對(duì)象字面量很酷

在對(duì)象字面量中只能使用__proto__一次,否則 JS 會(huì)報(bào)錯(cuò):

  1. var object = { 
  2.   __proto__: { 
  3.     toString: function() { 
  4.       return '[object Numbers]' 
  5.     } 
  6.   }, 
  7.   numbers: [1, 5, 89], 
  8.   __proto__: { 
  9.     toString: function() { 
  10.       return '[object ArrayOfNumbers]' 
  11.     } 
  12.   } 
  13. }; 

上面示例中的對(duì)象字面量中使用兩次__proto__屬性,這是不允許的。在這種情況下,將在會(huì)拋出錯(cuò)誤:SyntaxError: Duplicate __proto__ fields are not allowed in object literals 。

JS 約束只能用一個(gè)對(duì)象或 null 作為 __proto__ 屬性的值。任何使用原始類(lèi)型(字符串,數(shù)字,布爾值)或 undefined 類(lèi)型都將被忽略,并且不會(huì)更改對(duì)象的原型。

  1. var objUndefined = { 
  2.   __proto__: undefined, 
  3. }; 
  4. Object.getPrototypeOf(objUndefined); // => {} 
  5. var objNumber = { 
  6.   __proto__: 15, 
  7. }; 
  8. Object.getPrototypeOf(objNumber); // => {} 

對(duì)象字面量使用 undefined 和 數(shù)字 15 來(lái)設(shè)置 __proto__ 值。因?yàn)閮H允許將對(duì)象或 null 用作原型,所以__proto__值將被忽略,但 objUndefined 和objNumber 仍具有其默認(rèn)原型:純 JS 對(duì)象 {}, 。

當(dāng)然,嘗試使用基本類(lèi)型來(lái)設(shè)置對(duì)象的原型也會(huì)很奇怪。

當(dāng)對(duì)象字面具有計(jì)算結(jié)果為'proto'的字符串時(shí) {['__proto__']:protoObj },也要小心。以這種方式創(chuàng)建的屬性不會(huì)更改對(duì)象的原型,而只是使用鍵 '__proto__' 創(chuàng)建一個(gè)擁有的屬性

3. 簡(jiǎn)寫(xiě)方法定義

可以使用較短的語(yǔ)法在對(duì)象常量中聲明方法,以省略 function 關(guān)鍵字和 : 冒號(hào)的方式。這被稱(chēng)為簡(jiǎn)寫(xiě)方法定義。

接著,咱們使用簡(jiǎn)寫(xiě)的方法來(lái)定義一些方法:

  1. var collection = { 
  2.   items: [], 
  3.   add(item) { 
  4.     this.items.push(item); 
  5.   }, 
  6.   get(index) { 
  7.     return this.items[index]; 
  8.   }, 
  9. }; 
  10. collection.add(15); 
  11. collection.add(3); 
  12. collection.get(0); // => 15 

一個(gè)很好的好處是,以這種方式聲明的方法被命名為函數(shù),這對(duì)于調(diào)試目的很有用。從上面示例中執(zhí)行 collection.add.name 會(huì)返回函數(shù)名稱(chēng) “add”。

(1) super 的使用

JS 一個(gè)有趣的改進(jìn)是使用 super 關(guān)鍵字作為從原型鏈訪問(wèn)繼承的屬性的能力。看下面的例子:

為什么 JS 中的對(duì)象字面量很酷

calc 是 numbers 對(duì)象的原型。在 numbers 的 sumElements方法中,可以使用super關(guān)鍵字從原型訪問(wèn)方法:super.sumElements()

最終,super 是從對(duì)象原型鏈訪問(wèn)繼承的屬性的快捷方式。

在前面的示例中,可以嘗試直接執(zhí)行 calc.sumElements() 來(lái)調(diào)用原型,會(huì)報(bào)錯(cuò)。然而,super.sumElements() 可以正確調(diào)用,因?yàn)樗L問(wèn)對(duì)象的原型鏈。并確保原型中的sumElements() 方法使用 this.numbers 正確訪問(wèn)數(shù)組。

super 存在清楚地表明繼承的屬性將被使用。

(2) super 使用限制

super 只能在對(duì)象字面量的簡(jiǎn)寫(xiě)方法定義內(nèi)使用。

如果試圖從普通方法聲明{ name: function(){} } 訪問(wèn)它,JS 將拋出一個(gè)錯(cuò)誤:

為什么 JS 中的對(duì)象字面量很酷

方法 sumElements 被定義為一個(gè)屬性: sumElements: function(){…}。因?yàn)閟uper 只能在簡(jiǎn)寫(xiě)方法中使用,所以在這種情況下調(diào)用它會(huì)拋出 SyntaxError: 'super' keyword unexpected here。

此限制在很大程度上不影響對(duì)象字面量的聲明方式。由于語(yǔ)法較短,因此通常最好使用簡(jiǎn)寫(xiě)方法定義。

4. 計(jì)算屬性名

在 ES6 之前,對(duì)象初始化使用的是字面量的形式,通常是靜態(tài)字符串。要?jiǎng)?chuàng)建具有計(jì)算名稱(chēng)的屬性,就必須使用屬性訪問(wèn)器。

  1. function prefix(prefStr, name) { 
  2.   return prefStr + '_' + name; 
  3. var object = {}; 
  4. object[prefix('number', 'pi')] = 3.14; 
  5. object[prefix('bool', 'false')] = false; 
  6. object; // => { number_pi: 3.14, bool_false: false } 

當(dāng)然,這種定義屬性的方式是令人愉快的。

接著使用簡(jiǎn)寫(xiě)方式來(lái)改完上面的例子:

  1. function prefix(prefStr, name) { 
  2.   return prefStr + '_' + name; 
  3. var object = { 
  4.   [prefix('number', 'pi')]: 3.14, 
  5.   [prefix('bool', 'false')]: false, 
  6. }; 
  7. object; // => { number_pi: 3.14, bool_false: false } 

[prefix('number','pi')]通過(guò)計(jì)算 prefix('number', 'pi') 表達(dá)式(即'number_pi')來(lái)設(shè)置屬性名稱(chēng)。

相應(yīng)地,[prefix('bool', 'false')] 將第二個(gè)屬性名稱(chēng)設(shè)置為'bool_false'。

(1) symbol 作為屬性名稱(chēng)

symbol 也可以用作計(jì)算的屬性名稱(chēng)。只要確保將它們包括在方括號(hào)中即可:{[Symbol('name')]:'Prop value'}

例如,用特殊屬性 Symbol.iterator 并迭代對(duì)象自身的屬性名稱(chēng)。如下示例所示:

為什么 JS 中的對(duì)象字面量很酷

[Symbol.iterator]: function *() { } 定義一個(gè)屬性,該屬性用于迭代對(duì)象的自有屬性。展開(kāi)運(yùn)算符 [... object] 使用迭代器并返回自有的屬性的列表。

剩余和展開(kāi)屬性

剩余屬性允許從對(duì)象中收集在分配銷(xiāo)毀后剩下的屬性。

下面的示例在解構(gòu)對(duì)象之后收集剩余的屬性:

  1. var object = { 
  2.   propA: 1, 
  3.   propB: 2, 
  4.   propC: 3, 
  5. }; 
  6. let { propA, ...restObject } = object; 
  7. propA; // => 1 
  8. restObject; // => { propB: 2, propC: 3 } 

展開(kāi)屬性允許將源對(duì)象的自有屬性復(fù)制到對(duì)象文字面量中。在此示例中,對(duì)象字面量從源對(duì)象收集到對(duì)象的其他屬性:

  1. var source = { 
  2.   propB: 2, 
  3.   propC: 3, 
  4. }; 
  5. var object = { 
  6.   propA: 1, 
  7.   ...source, 
  8. }; 
  9. object; // => { propA: 1, propB: 2, propC: 3 } 

6. 總結(jié)

在 ES6 中,即使是作為對(duì)象字面量的相對(duì)較小的結(jié)構(gòu)也得到了相當(dāng)大的改進(jìn)。

可以使用__proto__ 屬性名稱(chēng)直接從初始化器設(shè)置對(duì)象的原型。這比使用Object.create() 更容易。

請(qǐng)注意,__proto__ 是 ES6 標(biāo)準(zhǔn)附件B的一部分,不鼓勵(lì)使用。該附件實(shí)現(xiàn)對(duì)于瀏覽器是必需的,但對(duì)于其他環(huán)境是可選的。NodeJS 4、5和6支持此功能。

現(xiàn)在方法聲明的形式更短,因此不必輸入 function 關(guān)鍵字。在簡(jiǎn)化方法中,可以使用super關(guān) 鍵字,該關(guān)鍵字可以輕松訪問(wèn)對(duì)象原型鏈中的繼承屬性。

如果屬性名稱(chēng)是在運(yùn)行時(shí)計(jì)算的,那么現(xiàn)在您可以使用計(jì)算的屬性名稱(chēng)[expression]來(lái)初始化對(duì)象。

 

責(zé)任編輯:趙寧寧 來(lái)源: 今日頭條
相關(guān)推薦

2020-03-31 16:30:09

JS語(yǔ)言ES 6

2021-10-09 07:10:30

Go字面量組合

2018-04-10 13:40:14

Kubernetes容器服務(wù)器

2020-12-18 11:35:22

TypeScript語(yǔ)言Java

2023-09-20 00:02:33

C++14二進(jìn)制字面量

2021-05-28 08:01:00

JS原型概念

2023-11-07 15:11:46

Kafka技巧

2022-06-08 08:01:28

模板字面量類(lèi)型

2009-07-06 14:42:24

Visual Basi

2020-08-02 22:54:04

Python編程語(yǔ)言開(kāi)發(fā)

2020-08-03 07:50:56

存儲(chǔ)對(duì)象存儲(chǔ)

2022-05-10 11:32:12

加密貨幣比特幣供應(yīng)限制

2022-05-10 14:17:26

加密貨幣供應(yīng)量以太坊

2015-02-11 10:22:25

對(duì)象存儲(chǔ)云共享S3存儲(chǔ)

2019-09-24 10:17:14

2012-05-02 15:56:20

PHP

2024-11-28 16:32:37

2023-09-19 08:03:01

JavaScriptevery()

2024-08-27 10:54:20

JSON函數(shù)屬性

2020-06-02 14:00:53

Vue.js組件Web開(kāi)發(fā)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)