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

資訊專欄INFORMATION COLUMN

ES6

AZmake / 541人閱讀

摘要:情況一情況二這兩種情況,根據(jù)的規(guī)定都是非法的。的作用域與命令相同只在聲明所在的塊級(jí)作用域內(nèi)有效。因此,將一個(gè)對(duì)象聲明為常量必須非常小心。頂層對(duì)象的屬性與全局變量掛鉤,被認(rèn)為時(shí)語(yǔ)言最大的設(shè)計(jì)敗筆之一。

這是ES6的入門篇教程的筆記,網(wǎng)址:鏈接描述,以下內(nèi)容中粗體+斜體表示大標(biāo)題,粗體是小標(biāo)題,還有一些重點(diǎn);斜體表示對(duì)于自身,還需要下功夫?qū)W習(xí)的內(nèi)容。這里面有一些自己的見解,所以若是發(fā)現(xiàn)問題,歡迎指出~
上一篇es5的到最后令人崩潰,看來深層的東西還是不太熟,希望這次不要這樣了?。。?/p>

ECMAScript 6簡(jiǎn)介
Babel轉(zhuǎn)碼器
以前構(gòu)建Vue-cli的時(shí)候一直不明白為什么要添加babel的依賴,現(xiàn)在才知道。。。。
Babel是一個(gè)廣泛使用的ES6轉(zhuǎn)碼器,可以將ES6代碼轉(zhuǎn)為ES5代碼,從而在現(xiàn)有環(huán)境執(zhí)行。這意味著,你可以用ES6的方式編寫程序,又不用擔(dān)心現(xiàn)有環(huán)境是否支持。

let 和 const 命令

1.let命令
基本用法
ES6新增了let命令,用來聲明變量。它的用法類似于var,但是所聲明的變量,只在let命令所在的代碼塊內(nèi)有效。

{
    let a = 10;
    var b = 1;
}
a // ReferenceError: a is not defined.
b // 1

// so for循環(huán)的計(jì)數(shù)器,就很合適使用let命令
for (let i = 0; i < 10; i++) {
    // ...
}
console.log(i); // ReferenceError: i is not defined

有一個(gè)重大發(fā)現(xiàn)!for循環(huán)還有一個(gè)特別之處,就是設(shè)置循環(huán)變量的那部分是一個(gè)父作用域,而循環(huán)體內(nèi)部是一個(gè)多帶帶的子作用域。這是以前沒注意的!!

for (let i = 0; i < 3; i++) {
    let i = "abc";
    console.log(i);
}
// abc
// abc
// abc 輸出3次abc,這表明函數(shù)內(nèi)部的變量i與循環(huán)變量i不在同一個(gè)作用域,有各自多帶帶的作用域。

不存在變量提升
var命令會(huì)發(fā)生“變量提升”現(xiàn)象,即變量可以在聲明之前使用,值為undefined;而let命令糾正了這種現(xiàn)象,它所聲明的變量一定要在聲明后使用,否則報(bào)錯(cuò)。

// var 的情況
console.log(foo); // 輸出undefined
var foo = 2;
// let 的情況
console.log(bar); // 報(bào)錯(cuò)ReferenceError
let bar = 2;

暫時(shí)性死區(qū)
只要塊級(jí)作用域內(nèi)存在let命令,它所聲明的變量就“綁定”(binding)這個(gè)區(qū)域,不再受外部的影響。
ES6明確規(guī)定,如果區(qū)塊中存在let和const命令,這個(gè)區(qū)塊對(duì)這些命令聲明的變量,從一開始就形成了封閉作用域。

var tmp = 123;
if (true) {
    tmp = "abc"; // ReferenceError
    let tmp; // 在塊級(jí)作用域內(nèi)又聲明了一個(gè)局部變量tmp,tmp綁定這個(gè)塊級(jí)作用域,凡是在聲明之前就使用這些變量,就會(huì)報(bào)錯(cuò)。
}

總之,在代碼塊內(nèi),使用let命令聲明變量之前,該變量都是不可用的,這在語(yǔ)法上,稱為“暫時(shí)性死區(qū)”(temporal dead zone, 簡(jiǎn)稱TDZ)。

if (true) {
    // TDZ開始
    tmp = "abc"; // ReferenceError
    console.log(tmp); // ReferenceError
    
    let tmp; // TDZ結(jié)束
    console.log(tmp); // undefined
    
    tmp = 123;
    console.log(tmp); // 123
}

不允許重復(fù)聲明
let不允許在相同作用域內(nèi),重復(fù)聲明同一個(gè)變量。

// 報(bào)錯(cuò)
function func() {
    let a = 10;
    var a = 1;
}
// 報(bào)錯(cuò)
function func() {
    let a = 10;
    let a = 1;
}

2.塊級(jí)作用域
為什么需要塊級(jí)作用域?
ES5只有全局作用域和函數(shù)作用域,沒有塊級(jí)作用域,這帶來很多不合理的場(chǎng)景。
第一種場(chǎng)景,內(nèi)層變量可能會(huì)覆蓋外層變量。

var tmp = new Date();
function f() {
    console.log(tmp); // 原意是調(diào)用外層的tmp變量
    if (false) {
        var tmp = "hello world";
    }
}
f(); // undefined 變量提升,導(dǎo)致內(nèi)層的tmp變量覆蓋了外層的tmp變量

第二種場(chǎng)景,用來計(jì)數(shù)的循環(huán)變量泄露為全局變量。

var s = "hello";
for (var i = 0; i < s.length; i++) {
    console.log(s[i]);
}
console.log(i); // 5 變臉i只用來控制循環(huán),但是循環(huán)結(jié)束后,它并沒有消失,泄露成了全局變量。

ES6的塊級(jí)作用域
let實(shí)際上為JavaScript新增了塊級(jí)作用域。

function f1() {
    let n = 5;
    if (true) {
        let n = 10;
    }
    console.log(n); // 5
}

塊級(jí)作用域與函數(shù)聲明
函數(shù)能不能在塊級(jí)作用域之中聲明?這是一個(gè)相當(dāng)令人混淆的問題。
ES5規(guī)定,函數(shù)只能在頂層作用域和函數(shù)作用域之中聲明,不能在塊級(jí)作用域聲明。

// 情況一
if (true) {
    function f() {}
}
// 情況二
try {
    function f() {}
} catch(e) {
    // ...
}
// 這兩種情況,根據(jù)ES5的規(guī)定都是非法的。

3.const命令
基本用法
const聲明一個(gè)只讀的常量。一旦聲明,常量的值就不能改變。
const聲明的變量不得改變值,這意味著,const一旦聲明變量,就必須立即初始化,不能留到以后賦值。

const foo; // SyntaxError: Missing initializer in const declaration 對(duì)于const來說,只聲明不賦值,就會(huì)報(bào)錯(cuò)。

const的作用域與let命令相同:只在聲明所在的塊級(jí)作用域內(nèi)有效。
const實(shí)際上保證的,并不是變量的值不得改動(dòng),而是變量指向的那個(gè)內(nèi)存地址所保存的數(shù)據(jù)不得改動(dòng)。對(duì)于簡(jiǎn)單類型的數(shù)據(jù)(數(shù)值、字符串、布爾值),值就保存在變量指向的那個(gè)內(nèi)存地址,因此等同于常量。但對(duì)于復(fù)合雷公的數(shù)據(jù)(主要是對(duì)象和數(shù)組),變量指向的內(nèi)存地址,保存的只是一個(gè)指向?qū)嶋H數(shù)據(jù)的指針,const只能保證這個(gè)指針是固定的(即總是指向另一個(gè)固定的地址),至于它指向的數(shù)據(jù)結(jié)構(gòu)是不是可變的,就完全不能控制了。因此,將一個(gè)對(duì)象聲明為常量必須非常小心。

const foo = {}; // foo儲(chǔ)存的是一個(gè)地址,這個(gè)地址指向一個(gè)對(duì)象,不可以變得只是這個(gè)地址,即不能把foo指向另一個(gè)地址,但對(duì)象本身是可變得,所以可以為其添加新屬性。
// 為foo添加一個(gè)屬性,可以成功
foo.prop = 123;
foo.prop // 123
// 將foo指向另一個(gè)對(duì)象,就會(huì)報(bào)錯(cuò)
foo = {}; // TypeError: "foo" is ready-only

如果真的想將對(duì)象凍結(jié),應(yīng)該使用Object.freeze方法。

const foo = Object.freeze({}); // 常量foo指向一個(gè)凍結(jié)得對(duì)象,所以下面得添加新屬性不起作用,嚴(yán)格模式時(shí)還會(huì)報(bào)錯(cuò)。
// 常規(guī)模式時(shí),下面一行不起作用;
// 嚴(yán)格模式時(shí),該行會(huì)報(bào)錯(cuò)
foo.prop = 123;
foo.prop // undefined

ES6聲明變量得六種方法
ES5只有兩種聲明變量得方法:var命令和function命令。ES6除了添加let和const命令,還有另外兩種聲明變量的方法:import命令和class命令。所以ES6一共有6種聲明變量的方法。

4.頂層對(duì)象的屬性
頂層對(duì)象,在瀏覽器環(huán)境指的是window對(duì)象,在Node指的是global對(duì)象。ES5之中,頂層對(duì)象的屬性與全局變量時(shí)等價(jià)的。
頂層對(duì)象的屬性與全局變量掛鉤,被認(rèn)為時(shí)JavaScript語(yǔ)言最大的設(shè)計(jì)敗筆之一。

window.a = 1;
a // 1
a = 2;
window.a // 2
// 以上代碼表示頂層對(duì)象的屬性賦值與全局變量的賦值,是同一件事。會(huì)帶來以下三個(gè)問題:
// 1)沒法再編譯時(shí)就報(bào)出變量未聲明的錯(cuò)誤,只有運(yùn)行時(shí)才能知道(因?yàn)槿肿兞靠赡苁琼攲觾冬F(xiàn)的屬性創(chuàng)造的,而屬性的創(chuàng)造是動(dòng)態(tài)的)。
// 2)程序員很容易不知不覺地就創(chuàng)建了全局變量
// 3)頂層對(duì)象的屬性是到處可以讀寫的,這非常不利于模塊化編程 
// 而且window對(duì)象有實(shí)體含義,指的是瀏覽器的窗口對(duì)象,頂層對(duì)象是一個(gè)有尸體含義的對(duì)象。

ES6為了保持兼容性,規(guī)定,var命令和function命令聲明的全局變量,依舊是頂層對(duì)象的屬性;另一方面規(guī)定,let命令、const命令、class命令聲明的全局變量,不屬于頂層對(duì)象的屬性。也就是說,從ES6開始,全局變量將逐步與頂層對(duì)象的熟悉那個(gè)脫鉤。

var a = 1;
window.a // 1
let b = 1;
window.b // undefined

變量的解構(gòu)賦值

1.數(shù)組的解構(gòu)賦值
基本用法
ES6允許按照一定模式,從數(shù)組和對(duì)象中提取值,對(duì)變量進(jìn)行賦值,這被成為解構(gòu)(Destructuring)。

let a = 1;
let b = 2;
let c = 3;
// 這是以前為變量賦值,只能直接指定值
// ES6允許寫成下面這樣
let [a, b, c] = [1, 2, 3]; // 表示可以從數(shù)組中提取值,按照對(duì)應(yīng)位置,對(duì)變量賦值。

上面這種寫法,本質(zhì)上,是屬于“模式匹配”,只要等號(hào)兩邊的模式相同,左邊的變量就會(huì)被賦予對(duì)應(yīng)的值。下面是一些使用嵌套數(shù)組進(jìn)行解構(gòu)的例子。

let [foo, [[bar], baz]] = [1, [[2], 3]];
foo // 1
bar // 2
baz // 3

let [,, third] = ["foo", "bar", "baz"];
third // "baz"

let [head, ...tail] = [1, 2, 3, 4];
head // 1
tail // [2, 3, 4]

let [x, y, ...z] = ["a"];
x // "a"
y // undefined
z // []

// 如果解構(gòu)不成功,變量的值就等于undefined
let [foo] = [];
let [bar, foo] = [1];
// 兩種情況都屬于解構(gòu)不成功,foo的值都會(huì)等于undefined

默認(rèn)值
解構(gòu)賦值允許指定默認(rèn)值。

let [foo = true] = [];
foo // true
let [x, y = "b"] = ["a"]; // x="a"",  y="b"
let [x, y = "b"] = ["a", undefined]; // x="a", y="b"

ES6內(nèi)部使用嚴(yán)格相等運(yùn)算符(===),判斷一個(gè)位置是否有值,所以,只有當(dāng)一個(gè)數(shù)組成員嚴(yán)格等于undefined。默認(rèn)值才會(huì)生效,如下:

let [x = 1] = [undefined];
x // 1
let [x = 1] = [null];
x // null 默認(rèn)值沒有生效,因?yàn)閚ull不嚴(yán)格等于undefined

如果默認(rèn)值是一個(gè)表達(dá)式,那么這個(gè)表達(dá)式是惰性求值的,即只有在用到的時(shí)候,才會(huì)求值。

function f() {
    console.log("aaa");
}
let [x = f()] = [1]; // 因?yàn)閤能取到值,所以函數(shù)f根本不會(huì)執(zhí)行,等價(jià)于下面的代碼

let x;
if ([1][0] === undefined) {
    x = f();
}
else {
    x = [1][0];
}

默認(rèn)值可以應(yīng)用解構(gòu)賦值的其他變量,但該變量必須已經(jīng)聲明。

let [x = 1, y = x] = []; // x=1; y=1
let [x = 1, y = x] = [2]; // x=2; y=2   x=2是因?yàn)閇2][0]不是undefined
let [x = 1, y = x] = [1, 2]; // x=1;y=2
let [x = y, y = 1] = []; // ReferenceError: y is not defined 因?yàn)閤用y做默認(rèn)值時(shí),y還沒有聲明

2.對(duì)象的解構(gòu)賦值
簡(jiǎn)介
解構(gòu)不僅可以用于數(shù)組,還可以用于對(duì)象。
對(duì)象的解構(gòu)與數(shù)組有一個(gè)重要的不同。數(shù)組的元素是按次序排列的,變量的取值由它的位置決定;而對(duì)象的屬性沒有次序,變量必須與屬性同名,才能取到正確的值。

let { bar, foo } = { foo: "aaa", bar: "bbb" };
foo // "aaa"
bar // "bbb"
let { baz } = { foo: "aaa", bar: "bbb" };
baz // undefined

對(duì)象的解構(gòu)賦值,可以很方便地將現(xiàn)有對(duì)象的方法,賦值到某個(gè)變量。

let { log, sin, cos } = Math; // 將Math對(duì)象的對(duì)數(shù)、正弦、余弦三個(gè)方法,賦值到對(duì)應(yīng)的變量上,使用起來就會(huì)方便很多
const { log } = console; // 將console.log賦值到log變量
log("hello") // hello

// 如果變量名與屬性名不一致,必須寫成下面這樣
let { foo: baz } = { foo: "aaa", bar: "bbb" };
baz // "aaa"
// 以上說明,對(duì)象的解構(gòu)賦值是下面形式的簡(jiǎn)寫。也就是說,對(duì)象的解構(gòu)賦值的內(nèi)部機(jī)制,是先找到同名屬性,然后再賦給對(duì)應(yīng)的變量。**真正被賦值的是后者,而不是前者。**
let { foo: foo, bar: bar} = { foo: "aaa", bar: "bbb"};
let { foo: baz } = { foo: "aaa", bar: "bbb" };
baz // "aaa" foo是匹配的模式,baz才是變量。真正被賦值的是變量baz,而不是模式foo。
foo // error: foo is not defined

// 與數(shù)組一樣,解構(gòu)也可以用于嵌套解構(gòu)的對(duì)象。
let obj = {
    p: [
        "Hello",
        { y: "World" }
    ]
};
let { p: [x, { y }] } = obj;
x // "Hello"
y // "World"
p // error: p is not defined 因?yàn)閜是模式,不是變量,不會(huì)被賦值

let obj = {
    p: [
        "Hello",
        { y: "World" }
    ]
};
let { p } = obj; // 這時(shí)p是變量
p // ["Hello", {y: "World"}]
x // error: x is not defined
y // error: y is not defined

// 如果想要p、x、y同時(shí)賦值
let obj = {
  p: [
    "Hello",
    { y: "World" }
  ]
};

let { p, p: [x, { y }] } = obj; // 第一個(gè)p是變量,可以將p賦值;第二個(gè)p是模式,將x、y賦值,因?yàn)閜已經(jīng)有值了,所以從p中獲?。ㄒ话闱闆r下,冒號(hào)‘:’表示模式)

如果解構(gòu)模式是嵌套的對(duì)象,而且子對(duì)象所在的父屬性不存在,那么將會(huì)報(bào)錯(cuò)。

// 報(bào)錯(cuò)
let {foo: {bar}} = {baz: "baz"} // 等號(hào)左邊對(duì)象的foo屬性,對(duì)應(yīng)一個(gè)子對(duì)象。該子對(duì)象的bar屬性,解構(gòu)時(shí)會(huì)報(bào)錯(cuò)。這是因?yàn)閒oo此時(shí)等于undefined,再取子屬性就會(huì)報(bào)錯(cuò)

注意點(diǎn)
(1)如果要將一個(gè)已經(jīng)聲明的變量用于解構(gòu)賦值,必須非常小心。

// 錯(cuò)誤的寫法
let x;
{x} = {x: 1} // SyntaxError: syntex error

代碼的寫法報(bào)錯(cuò),時(shí)因?yàn)镴avaScript引擎會(huì)將{x}理解成一個(gè)代碼塊,從而發(fā)生語(yǔ)法錯(cuò)誤。只有不將大括號(hào)寫在行首,避免JavaScript將其解釋為代碼塊,才能解決這個(gè)問題。

// 正確的寫法
let x;
({x} = {x: 1});

(2)解構(gòu)賦值允許等號(hào)左邊的模式之中,不防止任何變量名。因此,可以寫出非常古怪的賦值表達(dá)式。

({} = [true, false]);
({} = "abc");
({} = []);
// 上面的表達(dá)式雖然毫無(wú)意義,但是語(yǔ)法是合法的,可以執(zhí)行。

3.字符串的解構(gòu)賦值
字符串也可以解構(gòu)賦值,這是因?yàn)榇藭r(shí),字符串被轉(zhuǎn)換成了一個(gè)類似數(shù)組的對(duì)象。

const [a, b, c, d, e] = "hello";
a // "h"
b // "e"
c // "l"
d // "l"
e // "o"

類似數(shù)組的對(duì)象都有一個(gè)length屬性,因此還可以對(duì)這個(gè)屬性解構(gòu)賦值。

let {length: len} = "hello";
len // 5

4.數(shù)值和布爾值的解構(gòu)賦值
解構(gòu)賦值時(shí),如果等號(hào)右邊事數(shù)值和布爾值,則會(huì)先轉(zhuǎn)為對(duì)象。
解構(gòu)賦值的規(guī)則是,只要等號(hào)右邊的值不是對(duì)象或數(shù)組,就先將其傳為對(duì)象。由于undefined和null無(wú)法轉(zhuǎn)為對(duì)象,所以對(duì)它們進(jìn)行解構(gòu)賦值,都會(huì)報(bào)錯(cuò)。

let {toString: s} = 123;
s === Number.prototype.toString // true *那s可以干什么嗎?表示疑惑*

let { prop: x } = undefined; // TypeError
let { prop: y } = null; // TypeError

5.函數(shù)參數(shù)的解構(gòu)賦值
函數(shù)的參數(shù)也可以使用解構(gòu)賦值。

function add([x, y]) { // 表面上,該函數(shù)的參數(shù)是一個(gè)數(shù)組
    return x + y;
}
add([1, 2]); // 3 傳入?yún)?shù)的那一刻,數(shù)組參數(shù)就被解構(gòu)成變量x和y

// so on
[[1, 2], [3, 4]].map(([a, b]) => a + b); // [3, 7]

// 函數(shù)參數(shù)的解構(gòu)也可以使用默認(rèn)值
function move({x = 0, y = 0} = {}) { // 后面加"={}",就是防止什么都沒傳時(shí),默認(rèn)為{}
    return [x, y];
}
move({x: 3}); // [3, 0]
move({}); // [0, 0]
move();  // [0, 0]
// 不信看下面的,哈哈哈
function move({x, y} = {x: 0, y: 0}) { // 這是給move的參數(shù)指定默認(rèn)值,而不是為變量x和y指定默認(rèn)值
    return [x, y];
}
move({x: 3, y: 8}); // [3, 8]
move({x: 3}); // [3, undefined]
move({}); // [undefined, undefined]
move(); // [0, 0]
// undefined會(huì)觸發(fā)函數(shù)參數(shù)的默認(rèn)值
[1, undefined, 3].map((x = "yes") => x); // [1, "yes", 3]

6.圓括號(hào)問題
解構(gòu)賦值雖然很方便,但是解析起來并不容易。對(duì)于編譯器來說,一個(gè)式子到底是模式,還是表達(dá)式,沒有辦法從一開始就知道,必須解析到(或解析不到)等號(hào)才能知道。
由此帶來的問題市,如果模式中出現(xiàn)圓括號(hào)怎么處理。ES6的規(guī)則是,只要有可能導(dǎo)致解構(gòu)的歧義,就不得使用圓括號(hào)。
但是,這條規(guī)則實(shí)際上不那么容易辨別,處理起來相當(dāng)麻煩。因此, 建議只要有可能,就不要在模式中放置圓括號(hào)。

不能使用圓括號(hào)的情況
1)變量聲明語(yǔ)句

// 全部報(bào)錯(cuò) 它們都是變量聲明語(yǔ)句,模式不能使用圓括號(hào)。
let [(a)] = [1];
let {x: (c)} = {};
let ({x: c}) = {};
let {(x: c)} = {};
let {(x): c} = {};

2)函數(shù)參數(shù)
函數(shù)參數(shù)也屬于變量聲明,因此不能帶有圓括號(hào)。

// 報(bào)錯(cuò)
function f([(z)]) { return z; }
function f([z, (x)]) { return x; }

3)賦值語(yǔ)句的模式

// 報(bào)錯(cuò)
({ p: a}) = {p: 42};
([a]) = [5];
[({ p: a }), { x: c }] = [{}, {}];
// 但是這樣是可以的
({p: a} = {p: 42});

可以使用圓括號(hào)的情況只有一種:賦值語(yǔ)句的非模式部分,可以使用圓括號(hào)。
下面三行語(yǔ)句都可以正確執(zhí)行,因?yàn)槭紫人鼈兌际琴x值語(yǔ)句,而不是聲明語(yǔ)句;其次它們的圓括號(hào)都不熟模式的一部分。第一行語(yǔ)句中,模式是取數(shù)組的第一個(gè)成員,跟圓括號(hào)無(wú)關(guān);第二行語(yǔ)句中,模式是p,而不是d;第三行語(yǔ)句與第一行語(yǔ)句的性質(zhì)一致。

[(b)] = [3]; // 正確
({p: (d)} = {}); // 正確
({(parseInt.prop)] = [3]; // 正確

7.用途
變量的解構(gòu)賦值用途很多。
1)交換變量的值

let x = 1;
let y = 2;
[x, y] = [y, x]; // 這個(gè)是數(shù)組賦值,不像對(duì)象賦值一樣去找對(duì)應(yīng)的名字

2)從函數(shù)返回多個(gè)值
函數(shù)只能返回一個(gè)值,如果要返回多個(gè)值,只能將它們放在數(shù)組或?qū)ο罄锓祷亍?/p>

// 返回一個(gè)數(shù)組
function example() {
    return [1, 2, 3];
}
let [a, b , c] = example();

// 返回一個(gè)對(duì)象
function example() {
    return {
        foo: 1, 
        bar: 2
    };
}
let { foo, bar } = example();

3)函數(shù)參數(shù)的定義
解構(gòu)賦值可以方便地將一組參數(shù)與變量名對(duì)應(yīng)起來

// 參數(shù)是一組有次序的值
function f([x, y, z]) {...}
f([1, 2, 3]);

// 參數(shù)是一組無(wú)次序的值
function f({x, y, z}) {...}
f({z: 3, y: 2, x: 1});

4)提取JSON數(shù)據(jù)
解構(gòu)賦值對(duì)提取JSON對(duì)象中的數(shù)據(jù),尤其有用。

let jsonData = {
    id: 42,
    status: "OK",
    data: [867, 5309]
};
let { id, status, data: number } = jsonData;
console.log(id, status, number); // 42, "OK", [867, 5309]

5)函數(shù)參數(shù)的默認(rèn)值

jQuery.ajax = function (url, {
    async = true,
    beforeSend = function () {},
    cache = true,
    complete = function () {},
    crossDomain = false,
    global = true, 
    // ... more config
} = {}) {
    // ... do stuff
};

6)遍歷Map結(jié)構(gòu)
之前知道有map遍歷,現(xiàn)在又出來了Map結(jié)構(gòu),要好好區(qū)分,不然就暈了。
任何部署了Interator接口的對(duì)象,都可以用for...of循環(huán)遍歷。Map結(jié)構(gòu)原生支持Interator接口,配合變量的解構(gòu)賦值,獲取鍵名和鍵值就非常方便。
百度了一下,map與其他鍵值對(duì)集合的區(qū)別,發(fā)現(xiàn)map的“key”范圍不僅限于字符串,而是各種類型的值都可以當(dāng)作key。也就是說,object提供了“字符串-值”的對(duì)應(yīng)結(jié)構(gòu),map則提供的是“值-值”的對(duì)應(yīng),是一種更加完善的hash結(jié)構(gòu)。

const map = new Map();
map.set("first", "hello");
map.set("second", "world");
for (let [key, value] of map) {
    console.log(key + " is " + value);
}
// first is hello
// second is world
// 如果只想獲取鍵名,或者只想獲取鍵值,可以寫成下面這樣
// 獲取鍵名
for (let [key] of map) {
    // ...
}
for (let [, value] of map) {
    // ...
}

7)輸入模塊的指定方法
加載模塊時(shí),往往需要指定輸入哪些方法。結(jié)構(gòu)賦值使得輸入語(yǔ)句非常清晰。

const { SourceMapConsumer, SourceNode } = require("source-map");

字符串的擴(kuò)展

1.字符串的Unicode表示法
emm感覺用Unicode表示一個(gè)字符的用處不大,所以跳過這節(jié)吧~

2.字符串的遍歷器接口
字符串可以被for...of循環(huán)遍歷,這個(gè)遍歷器最大的優(yōu)點(diǎn)是可以識(shí)別大于0xFFFF的碼點(diǎn),傳統(tǒng)的for循環(huán)無(wú)法識(shí)別這樣的碼點(diǎn)。

for (let codePoint of "foo") {
    console.log(codePoint)
}
// "f"
// "o"
// "o"

5.模板字符串
模板字符串(template string)是增強(qiáng)版的字符串,用反引號(hào)(`)表示。它可以當(dāng)作普通字符串使用,也可以用來定義多行字符串,或者在字符串中嵌入變量。模板字符串中嵌入變量,需要將變量名寫在${}之中。


大括號(hào)內(nèi)部可以放入任意的JavaScript表達(dá)式,可以進(jìn)行運(yùn)算,以及引用對(duì)象屬性。模板字符串之中還能調(diào)用函數(shù)。

let x = 1;
let y = 2;
`${x} + ${y} = ${x + y}` // "1 + 2 = 3"

let obj = {x: 1, y: 2};
`${obj.x + obj.y}` // "3"

function fn() {
    return "Hello World";
}
`foo ${fn()} bar` // foo Hello World bar

如果需要引用模板字符串本身,在需要時(shí)執(zhí)行,可以寫成函數(shù)。模板字符串寫成了一個(gè)函數(shù)的返回值。執(zhí)行這個(gè)函數(shù),就相當(dāng)于執(zhí)行這個(gè)模板字符串了。

let func = (name) => `Hello ${name}!`;
func("Jack") // "Hello Jack!"

emmm 下面的看不懂了,也沒在代碼中看到過,看來后期的學(xué)習(xí)還有很長(zhǎng)一段路呢。

字符串的新增方法
前面的那幾種方法感覺并不常用,只是粗略看了一遍,并沒有細(xì)看。

6.實(shí)例方法:repeat()
repeat方法返回一個(gè)新字符串,表示將原字符串重復(fù)n次。

"hello".repeat(2) // "hellohello"
"na".repeat(0) // ""
"na".repeat(2.9) // "nana" 小數(shù)會(huì)被取整,向下取整

正則的擴(kuò)展

1、RegExp構(gòu)造函數(shù)
在ES5中,RegExp構(gòu)造函數(shù)的參數(shù)有兩種情況。
第一種情況是,參數(shù)是字符串,這時(shí)第二個(gè)參數(shù)表示正則表達(dá)式的修飾符(flag)。
第二種情況是,參數(shù)是一個(gè)正則表達(dá)式,這時(shí)會(huì)返回一個(gè)原有正則表達(dá)式的拷貝。

//  第一種
let regex = new RegExp("xyz", "i");
// 等價(jià)于
let regex = /xyz/i;

// 第二種
let regex = new RegExp(/xyz/i);
// 等價(jià)于
let regex = /xyz/i;

// 但是不允許此時(shí)使用第二個(gè)參數(shù)添加修飾符,否則會(huì)報(bào)錯(cuò)。
let regex = new RegExp(/xyz/, "i"); // Uncaught TypeError: Cannot supply flags when constructing one RegExp from another.

2、字符串的正則方法
字符串對(duì)象共有4個(gè)方法,可以使用正則表達(dá)式:match()、replace()、search()、split()。
ES6將這4個(gè)方法,在語(yǔ)言內(nèi)部全部調(diào)用RegExp的實(shí)例方法,從而做到所有與正則相關(guān)的方法,全部定義在RegExp對(duì)象上。

 - String.prototype.match調(diào)用RegExp.prototype[Symbol.match]
 - String.prototype.replace調(diào)用RegExp.prototype[Symbol.replace]
 - String.prototype.search調(diào)用RegExp.prototype[Symbol.search]
 - String.prototype.split調(diào)用RegExp.prototype[Symbol.split]

數(shù)值的擴(kuò)展

2、Number.isFinite(),Number.isNaN()
ES6在Number對(duì)象上,新提供了Number.isFinite()和Number.isNaN()兩個(gè)方法。
Number.isFinite()用來檢查一個(gè)數(shù)值是否為有限的(finite),即不是Infinity。
注:如果參數(shù)類型不是數(shù)值,Number.isFinite一律返回false。

Number.isFinite(0.8); // true
Number.isFinite(NaN); // false
Number.isFinite(Infinity); // false
Number.isFinite(-Infinity); // false
Number.isFinite("foo"); // false
Number.isFinite("15"); // false
Number.isFinite(true); // false

Number.isNaN()用來檢查一個(gè)值是否為NaN。
注:如果參數(shù)類型不是數(shù)值,Number.isFinite一律返回false。

Number.isNaN(NaN); // true
Number.isNaN(15); // true
Number.isNaN("15"); // false
Number.isNaN(true); // false
Number.isNaN(9 / NaN); // true
Number.isNaN("true" / 0); // true
Number.isNaN("true" / "true"); // true

它們與傳統(tǒng)的全局方法isFinite()和isNaN()的區(qū)別在于,傳統(tǒng)方法先調(diào)用Number()將非數(shù)值的值轉(zhuǎn)為數(shù)值,在進(jìn)行判斷,而這兩個(gè)新方法只對(duì)數(shù)值有效,Number.isFinite()對(duì)于非數(shù)值一律返回false,Number.isNaN()只有對(duì)于NaN才返回true,非NaN一律返回false。

isFinite("25"); // true
Number.isFinite("25"); // false

isNaN("NaN"); // false
Number.isNaN("NaN"); // true

3、Number.parseInt(),Number.parseFloat()
ES6將全局方法parseInt()和parseFloat(),移植到Number對(duì)象上面,行為完全保持不變。
這樣做的目的,是逐步減少全局性方法,使得語(yǔ)言逐步模塊化。

// ES5的寫法
parseInt("12.34") // 12
parseFloat("123.45#") // 123.45

// ES6的寫法
Number.parseInt("12.34") // 12
Number.parseFloat("12.345#") // 123.45

4、Number.isInteger()
Number.isInteger()用來判斷一個(gè)數(shù)值是否為整數(shù),也就是說,是直接判斷的。

Number.isInteger(25); // true
Number.isInteger(25.1); // false
Number.isInteger(); // false
Number.isInteger(null); // false
Number.isInteger("25"); // false
Number.isInteger(true); // false

5、Number.EPSILON
ES6在Number對(duì)象上面,新增一個(gè)極小的常量Number.EPSILON。根據(jù)規(guī)格,它表示1與大于1的最小浮點(diǎn)數(shù)之間的差。(相當(dāng)于2的-52次方。)
Number.EPSILON實(shí)際上是JavaScript能夠表示的最小精度。誤差如果小于這個(gè)值,就可以認(rèn)為已經(jīng)沒有意義了,即不存在誤差了。
Number.EPSILON可以用來設(shè)置“能夠接受的誤差范圍”。比如。誤差范圍設(shè)為2的-50次方(即Number.EPSILON * Math.pow(2, 2)),即如果兩個(gè)浮點(diǎn)數(shù)的差小于這個(gè)值,我們就認(rèn)為這兩個(gè)浮點(diǎn)數(shù)相等。

function withinErrorMargin (left, right) {
    return Math.abs(left - right) < Number.EPSILON * Math.pow(2, 2);
}
0.1 + 0.2 === 0.3; // false
withinErrorMargin(0.1 + 0.2, 0.3); // true

7、Math對(duì)象的擴(kuò)展
ES6在Math對(duì)象上新增了17個(gè)與數(shù)學(xué)相關(guān)的方法。所有這些方法都靜態(tài)方法,只能在Math對(duì)象上調(diào)用。

Math.trunc()
Math.trunc方法用于去除一個(gè)數(shù)的小數(shù)部分,返回整數(shù)部分。
對(duì)于非數(shù)值,Math.trunc內(nèi)部使用Number方法將其先轉(zhuǎn)為數(shù)值。

Math.trunc(4.1); // 4
Math.trunc(4.9); // 4
Math.trunc(-4.9); // -4
Math.trunc("123.456"); // 123
Math.trunc(true); // 1
Math.trunc(null); // 0
Math.trunc(NaN); // NaN
Math.trunc("foo"); // NaN
Math.trunc(undefined); // NaN

// 實(shí)際意義
Math.trunc = Math.trunc || function(x) {
    return x < 0 ? Math.ceil(x) : Math.floor(x);
}

Math.sign()
Math.sign()方法用來判斷一個(gè)數(shù)到底是正數(shù)、負(fù)數(shù)、還是零。對(duì)于非數(shù)值,會(huì)先將其轉(zhuǎn)換為數(shù)值。
它會(huì)返回五種值:

參數(shù)為正數(shù),返回+1; Math.sign(5) // +1

參數(shù)為負(fù)數(shù),返回-1; Math.sign(-5) // -1

參數(shù)為0, 返回0; Math.sign(0) // +0

參數(shù)為-0,返回-0; Math.sign(-0) // -0

其他值,返回NaN。 Math.sign(NaN) // NaN

如果參數(shù)是非數(shù)值,會(huì)自動(dòng)轉(zhuǎn)為數(shù)值。對(duì)于那些無(wú)法轉(zhuǎn)為數(shù)值的值,會(huì)返回NaN。

Math.sign("") // 0
Math.sign(true) // +1
Math.sign(false) // 0
Math.sign(null) // 0
Math.sign("foo") // NaN
Math.sign("9") // +1
Math.sign() // NaN
Math.sign(undefined) // NaN

Math.cbrt()
Math.cbrt方法用于計(jì)算一個(gè)數(shù)的立方根。
對(duì)于非數(shù)值,Math.cbrt方法內(nèi)部也是先使用Number方法將其轉(zhuǎn)為數(shù)值。

Math.cbrt(-1) // -1
Math.cbrt(2) // 1.2599210498948734
Math.cbrt("8") // 2
Math.cbrt("hello") // NaN

// 實(shí)質(zhì)
Math.cbrt = Math.cbrt || function(x) {
    let y = Math.power(Math.abs(x), 1/3);
    return x < 0 ? -y : y;
}

8、指數(shù)運(yùn)算符
ES2016新增了一個(gè)指數(shù)運(yùn)算符(**)。該運(yùn)算符是右結(jié)合,而不是常見的左結(jié)合。多個(gè)指數(shù)運(yùn)算符連用時(shí),是從最右邊開始計(jì)算的。

2 ** 3 // 8
2 ** 3 ** 2 // 512 相當(dāng)于 2**(3**2)

指數(shù)運(yùn)算符可以與等號(hào)結(jié)合,形成一個(gè)新的賦值運(yùn)算符(**=)。

a **= 2 // 等同于 a = a * a
b **= 3 // 等同于 b = b * b *  b

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/105438.html

相關(guān)文章

  • 使用ES6新特性開發(fā)微信小程序

    摘要:使用新特性開發(fā)微信小程序國(guó)際化與本地化新特性國(guó)際化與本地化新增了很多對(duì)于國(guó)際化的支持,比如時(shí)間格式,貨幣格式,數(shù)字格式等。 ECMAScript 6(簡(jiǎn)稱ES6)是JavaScript語(yǔ)言的最新標(biāo)準(zhǔn)。因?yàn)楫?dāng)前版本的ES6是在2015年發(fā)布的,所以又稱ECMAScript 2015。 微信小程序支持絕大部分ES6的新增特性。 使用ES6新特性開發(fā)微信小程序(1) ES6新特性:Cons...

    Backache 評(píng)論0 收藏0
  • ES6-前世今生(0)

    摘要:更新了個(gè)版本,最新正式版是語(yǔ)言的下一代標(biāo)準(zhǔn),早已在年月正式發(fā)布?;静恢С忠苿?dòng)端瀏覽器對(duì)的支持情況版起便可以支持的新特性。比較通用的工具方案有,,,等。 1、ECMAScript是什么? 和 JavaScript 有著怎樣的關(guān)系? 1996 年 11 月,Netscape 創(chuàng)造了javascript并將其提交給了標(biāo)準(zhǔn)化組織 ECMA,次年,ECMA 發(fā)布 262 號(hào)標(biāo)準(zhǔn)文件(ECMA-...

    LeviDing 評(píng)論0 收藏0
  • 給React初學(xué)者的10分鐘ES6教程

    摘要:但是在中,可以通過關(guān)鍵字來實(shí)現(xiàn)類的繼承的使用可以使得繼承意義更加明確并且值得一提的是,如果你使用來定義的組件,那么可以在類的構(gòu)造器里面,用簡(jiǎn)單的的聲明方式來替代方法。 原文:The 10 min ES6 course for the beginner React Developer譯者:Jim Xiao 著名的80/20定律可以用來解釋React和ES6的關(guān)系。因?yàn)镋S6增加了超過75...

    Awbeci 評(píng)論0 收藏0
  • 10個(gè)最佳ES6特性

    摘要:,正式名稱是,但是這個(gè)名稱更加簡(jiǎn)潔。已經(jīng)不再是最新的標(biāo)準(zhǔn),但是它已經(jīng)廣泛用于編程實(shí)踐中。而制定了模塊功能。自從年雙十一正式上線,累計(jì)處理了億錯(cuò)誤事件,得到了金山軟件等眾多知名用戶的認(rèn)可。 譯者按: 人生苦短,我用ES6。 原文: Top 10 ES6 Features Every Busy JavaScript Developer Must Know 譯者: Fundebug 為了保...

    codeKK 評(píng)論0 收藏0
  • 為什么都說js 里面任何對(duì)象最終都繼承了Object對(duì)象

    摘要:今天閑來無(wú)事,看見幾行小字。又說所有對(duì)象,繼承終是。強(qiáng)行押韻一波這首詩(shī)的意思就是說的我今天沒有什么事情,然后無(wú)意中又在網(wǎng)上看到了任何對(duì)象都是從對(duì)象繼承而來的這句話。一時(shí)興起,便去驗(yàn)證這句話。 今天閑來無(wú)事,看見幾行小字。又說所有對(duì)象,繼承終是Obj?!?強(qiáng)行押韻一波 這首詩(shī)的意思就是說的我今天沒有什么事情,然后無(wú)意中又在網(wǎng)上看到了任何對(duì)象都是從Object對(duì)象繼承而來的這句話。一時(shí)興...

    Gemini 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<