ECMAScript 之建立 Object

栏目: JavaScript · 发布时间: 6年前

内容简介:ECMAScript 也有 Object 概念,但其 Object 與傳統 OOP 的 Object 又不太一樣。傳統 OOP 的 Object 都必須由 class 而來,但 ECMAScript 的 Object 卻有多種建立方式。有別於傳統 OOP 一定要先建立 class,才能使用不需要建立 class,可以直接以 declarative 方式建立 object。

ECMAScript 也有 Object 概念,但其 Object 與傳統 OOP 的 Object 又不太一樣。傳統 OOP 的 Object 都必須由 class 而來,但 ECMAScript 的 Object 卻有多種建立方式。

建立 Object

有別於傳統 OOP 一定要先建立 class,才能使用 new 建立 object,ECMAScript 提供了 6 種建立 object 方式:

  • Object Literal
  • Empty Object
  • New Object
  • Constructor Function
  • Class
  • Object.create()

Object Literal

不需要建立 class,可以直接以 declarative 方式建立 object。

var person = {
  firstName: 'Sam',
  lastName: 'Xiao',
  fullName: function() {
    return this.firstName + ' ' + this.lastName;
  }  
};

console.log(person);
console.log(person.fullName());

以 key-value 方式直接在 {} 之間建立 object,其中 fullName 相當於傳統 OOP 的 method,但是將 function 指定給 fullName property。

ECMAScript 之建立 Object

  1. 直接將 function 指定給 property
  2. 因為 person 為 object,可如同 OOP 般使用 method 方式使用 fullName()

Empty

ECMAScript 有一個傳統 OOP 沒有的特色,就是可以動態新增 property,既然如此,我們就可以一開始只宣告一個 空物件{} ,事後再動態新增 property 即可。

var person = {};
person.firstName = 'Sam';
person.lastName = 'Xiao';
person.fullName = function() {
  return this.firstName + ' ' + this.lastName;
};

console.log(person);
console.log(person.fullName());

一開始只要指定 {} 即可,表示其為 Object type,事後再動態新增 property。

ECMAScript 之建立 Object

  1. 一開始只要宣告為 {} 即可,不需要另外建立 class,也不需要 new ,事後再動態建立 property
  2. 因為 person 為 object,可如同 OOP 般使用 method 方式使用 fullName()

New Object

var person = new Object();
person.firstName = 'Sam';
person.lastName = 'Xiao';
person.fullName = function() {
  return this.firstName + ' ' + this.lastName;
};

console.log(person);
console.log(person.fullName());

由於 Object 為 ECMAScript 內建型別之一,因此你也可以使用 new Object() 方式建立 object。

不過實務上幾乎不會這樣寫,因為 let person = {} 寫法更精簡

ECMAScript 之建立 Object

  1. 也可以使用 new Object() 建立 object,但 WebStorm 已經提出建議: 是否要重構成 object literal ,也就是使用 Empty Object 方式。

Constructor Function

以上 3 種方式適合需要以 declarative 方式建立 object 時使用,也就是 data 自己可以決定,但實務上 data 幾乎由 user 決定,也就是要以 parameter 傳入。

另外這 3 種方式,每次建立 object 時,都會重新建立 function,然後指定到 fullName property,事實上我們發現,每個 object 只有 data 會不一樣,但 function 都是一樣的,因此不斷地建立 相同的 funtion 很浪費的記憶體。

比較好的方式是 function 只建立一次,然後各 object 共用相同的 function。

function Person(firstName, lastName) {
  this.firstName = firstName;
  this.lastName = lastName;
}

Person.prototype.fullName = function() {
  return this.firstName + ' ' + this.lastName;
};

var person = new Person('Sam', 'Xiao');
console.log(person);
console.log(person.fullName());

由於 ECMAScript 本來就是以 function 為主的語言,因此使用 function 建立 object 也就不意外了。

其中 this 所建立的 property 都是 public

為了避免 function 重複建立,在 Person constructor function 的 prototype 指定 fullName property 為 fuction,也就是儘管建立多個 object,但 fullName 都指向同一個 function,因此節省記憶體。

是否發現這種寫法,似乎就是傳統 OOP 的 Class + Constructor 合體呢 ?

ECMAScript 之建立 Object

new

Class

class Person {
  constructor(firstName, lastName) {
    this.firstName = firstName;
    this.lastName = lastName;
  }

  fullName() {
    return this.firstName + ' ' + this.lastName;
  }
}

var person = new Person('Sam', 'Xiao');
console.log(person);
console.log(person.fullName());

ECMAScript 2015 新增了 classconstructor ,讓我們可以類似傳統 OOP 使用 class ,但其本質其實是 ES5 的 Constructor Function,是道地的 syntax sugar。

與傳統 OOP 不同的是:ECMAScript 2015 的 class 並不用宣告 public property (也不能宣告 property,因為不合語法),而是直接在 constructor() 內使用 this 定義 property,因為其本質仍是 ES5 的 Constructor Function。

ECMAScript 之建立 Object

this
new

雖然 fullName() 看起來是寫在 class 內,但仍然是透過 prototype 建立,我們可以使用 TypeScript Playground 獲得證明:

ECMAScript 之建立 Object

fullName()

Object.create()

var prototype = {
  fullName: function() {
    return this.firstName + ' ' + this.lastName;
  },
};

var person = Object.create(prototype);
person.firstName = 'Sam';
person.lastName = 'Xiao';

console.log(person);
console.log(person.fullName());

Object.create() 也可以建立 object,它提供了兩個參數。

  • 第一個參數:指定新 object 的 prototype object
  • 第二個參數:指定新 object 的 property object,可省略

由於 ECMAScript 可以動態新增 property,因此可以先不指定 property object,事後再動態新增 property。

但 method 部分會建立在 prototype,因此先將 fullName() 建立在 prototype object,再透過 Object.create() 建立,並將 prototype object 傳入。

ECMAScript 之建立 Object

  1. 建立 prototype object,並將 method 定義在 prototype 內
  2. 將 prototype object 傳進 Object.create() ,表示 personfullName() 來自於 prototype object
  3. Property 動態建立即可
var prototype = {
  fullName: function() {
    return this.firstName + ' ' + this.lastName;
  },
};

var property = {
  firstName: {
    value: 'Sam',
    writable: true,
    configurable: true,
  },
  lastName: {
    value: 'Xiao',
    writable: true,
    configurable: true,
  },
};

var person = Object.create(prototype, property);

console.log(person);
console.log(person.fullName());

Object.create() 還有第二種用法,將 property 定義在第二個參數 property object,但 method 一樣定義在 prototype object。

ECMAScript 之建立 Object

Object.create()
{}

ECMAScript 之建立 Object

  1. 在 Firefox 中可看到 person 完整印出,這才是對的

Conclusion

  • Object Literal、Empty Object 或 New Object 方式,雖然可以建立 method,但只能建立在 object 上,較浪費記憶體
  • Constructor Function、Class 或 Object.create(),則能夠將 method 建立在 object 的 prototype 上,較節省記憶體
  • Class 本質上仍然是 Constructor Function,可由 TypeScript 與 Babel 得證

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

C++标准模板库编程实战

C++标准模板库编程实战

Ivor Horton / 郭小虎、程聪 / 2017-1

《C++标准模板库编程实战》介绍最新的C++14标准的API、库和扩展,以及如何将它们运用到C++14程序中。在书中,作者Ivor Horton 则阐述了什么是STL,以及如何将它们应用到程序中。我们将学习如何使用容器、迭代器,以及如何定义、创建和应用算法。此外,还将学习函数对象和适配器,以及它们的用法。 阅读完本书之后,你将能够了解如何扩展STL,如何定义自定义类型的C++组件,你还将能够......一起来看看 《C++标准模板库编程实战》 这本书的介绍吧!

JS 压缩/解压工具
JS 压缩/解压工具

在线压缩/解压 JS 代码

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具

UNIX 时间戳转换
UNIX 时间戳转换

UNIX 时间戳转换