成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

ECMAScript 6新特性印象之二:面對對象和模塊化

darryrzhong / 3097人閱讀

摘要:本文參考了以下文章之前的文章新特性印象之一新語法面對對象關(guān)鍵字看上面例子就能明白。定義類的,配合創(chuàng)建新對象。繼承非構(gòu)造器對象的原型是。錯誤檢查繼承的目標(biāo)一定要是個對象或者。的構(gòu)造器是可改寫,但不可枚舉。引入了一個標(biāo)簽,負(fù)責(zé)載入模塊。

本文參考了以下文章/PPT:

Use ECMAScript 6 today

Ecmascript 6 Whats next for Javascript

ECMAScript 6: classes

ECMAScript 6 modules: the final syntax

es6features

Javascript Modules

之前的文章:

ECMAScript 6新特性印象之一:新語法

面對對象

1.關(guān)鍵字 Class

class Artist {
    constructor(name) {
        this.name = name;
    }
    perform() {
        return this.name + " performs ";
    }
}

class Singer extends Artist {
    constructor(name, song) {
        super.constructor(name);
        this.song = song;
    }
    perform() {
        return super.perform() + "[" + this.song + "]";
    }
}

let james = new Singer("Etta James", "At last");
james instanceof Artist; // true
james instanceof Singer; // true

james.perform(); // "Etta James performs [At last]"

看上面例子就能明白。注意幾個關(guān)鍵字extendssuper。

雖然ES6的Class本質(zhì)上還是語法糖,但這么設(shè)計有它的目的。

在ES5中, function關(guān)鍵字承擔(dān)著三個職責(zé):

定義函數(shù)。

定義方法屬性。

定義類的constructor,配合new創(chuàng)建新對象。

在ES6中,第2點(diǎn)被屬性方法定義(Method definitions)替代,第3點(diǎn)被Class關(guān)鍵字替代。一個關(guān)鍵字只承擔(dān)一個職責(zé),不再是滿屏function,足夠清晰了吧?

有幾點(diǎn)要注意的:

類的body只能包含屬性方法,不能包含屬性值。屬性值放到constructor方法里。

屬性方法可以是生成器,在方法名前家*就可以。

聲明類(Class Declaration)并不會被提升(hoisted)。

如果沒有指定constructor,那會有個默認(rèn)調(diào)用super的。

2.繼承 Extending

繼承的幾種情況和規(guī)則:

不要繼承空類, class Foo {},因?yàn)椋?/p>

Foo的原型(prototype)是Function.prototype(所有函數(shù)的原型都是這個)。

而Foo.prototype的原型是Object.prototype

這種繼承就和函數(shù)一樣了。

繼承nullclass Foo extends null {}

Foo的原型是Function.prototype。

Foo.prototype的原型是null

這樣Object.prototype的屬性方法都不會繼承到Foo中。

繼承構(gòu)造器:class Foo extends SomeClass

Foo的原型是SomeClass

Foo.prototype的SomeClass.prototype。

這樣,類方法屬性也會被繼承。

繼承非構(gòu)造器(對象):class Foo extends SomeObject

Foo的原型是Function.prototype

Foo.prototype的SomeClass。

錯誤檢查:繼承的目標(biāo)一定要是個對象或者null。如果是繼承構(gòu)造器,那么構(gòu)造器的原型一定要是個對象或者null。

類聲明其實(shí)創(chuàng)建的是可變let綁定(binding,函數(shù)式編程會比較熟悉這個概念)。對于一個類Foo:

Foo的原型是不可改寫,且不可枚舉的。

Foo的構(gòu)造器是可改寫,但不可枚舉。

Foo的原型函數(shù)(Foo.prototype.*)是可改寫,但不可枚舉。

模塊

ES6的內(nèi)置模塊系統(tǒng)借鑒了CommonJS和AMD各自的優(yōu)點(diǎn):

具有CommonJS的精簡語法、唯一導(dǎo)出出口(single exports)和循環(huán)依賴(cyclic dependencies)的特點(diǎn)。

類似AMD,支持異步加載和可配置的模塊加載。

不僅僅借鑒,ES6還對這些功能進(jìn)行了優(yōu)化:

語法比CommonJS更精簡。

支持結(jié)構(gòu)的靜態(tài)分析(靜態(tài)檢查,優(yōu)化等等)。

循環(huán)依賴的支持比CommonJS更好。

1.語法

ES6模塊的導(dǎo)出出口形式有兩種:命名導(dǎo)出(一個模塊多個導(dǎo)出)、標(biāo)準(zhǔn)導(dǎo)出(一個模塊一個導(dǎo)出)。

命名導(dǎo)出 Named exports

//-------lib.js----------
export const sqrt = Math.sqrt;
export function square(x) {
    return x * x;
}
export function diag(x, y) {
    return sqrt(square(x) + square(y));
}

//-------main1.js--------
import { sqaure, diag } from "lib";

console.log(square(11)); // 121
console.log(diag(3,4)); // 5

//或者這樣,命名導(dǎo)入的名稱:
//-------main2.js--------
import * as lib from "lib";
console.log(lib.square(11)); // 121
console.log(lib.diag(3,4)); // 5

標(biāo)準(zhǔn)導(dǎo)出 Default exports

//-------MyClass.js-----
// 注意,`export`的操作對象是表達(dá)式,一般是沒有名字的。
export default class { ... };

//-------main3.js--------
// 通過模塊名字識別
import MyClass from "MyClass";
let inst = new MyClass();

當(dāng)然,這兩種導(dǎo)出也可以混合使用。本質(zhì)上,標(biāo)準(zhǔn)導(dǎo)出只是指名導(dǎo)出的一種,名稱是「default」而已。

就目前了解的來看,ES6的模塊導(dǎo)出貌似有些繁瑣,還不如CommonJS直接通過object導(dǎo)出利索。

2.設(shè)計初衷 Design goals

TC39在討論ES6的模塊化問題時,主要考慮了下面幾點(diǎn):

直接導(dǎo)出優(yōu)先

靜態(tài)模塊結(jié)構(gòu)

同步/異步載入都要支持

支持循環(huán)依賴

第一點(diǎn)就是要簡單已用。而靜態(tài)模塊結(jié)構(gòu)更多是出于性能優(yōu)化、靜態(tài)類型檢查(沒錯,就是這個,動態(tài)語言為什么要有這個,其實(shí)還是為了性能)和以后的可能會加入的「宏模板」功能。

3.更多導(dǎo)入/導(dǎo)出寫法舉例

導(dǎo)入:

// 標(biāo)準(zhǔn)導(dǎo)入,命名導(dǎo)入
import theDefault, { named1, named2 } from "src/mylib";
import theDefault from from "src/mylib";
import { named1, named2 } from "src/mylib";

// 重命名
import { named1 as myNamed1, named2 } from "src/mylib";

// 將導(dǎo)入的模塊定義為一個對象
// 模塊的每個屬性都是該對象的同名方法屬性
import * as mylib from "src/mylib";

// 僅讀取模塊,不導(dǎo)入任何功能
import "src/mylib";

導(dǎo)出:

// 使用關(guān)鍵字**export**導(dǎo)出
export let myVar1 = ...;
export function MyFunc() {...}
export function* myGeneratorFunc() {...}

// 也可以以對象形式導(dǎo)出
const MY_CONST = ...;
function myFunc() { ... }

export { MY_CONST, myFunc }
// 當(dāng)然,名字不一定要相同
export { MY_CONST as THE_CONST, myFunc as theFunc };

// 支持重導(dǎo)出,即在當(dāng)前模塊導(dǎo)出其他模塊的功能 方便hack
export * from "src/other_module";
export { foo, bar } from "src/other_module";

上面說的這些語法,普通的

閱讀需要支付1元查看
<