摘要:情況一情況二這兩種情況,根據(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
摘要:使用新特性開發(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...
摘要:更新了個(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-...
摘要:但是在中,可以通過關(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...
摘要:,正式名稱是,但是這個(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 為了保...
摘要:今天閑來無(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í)興...
閱讀 3291·2021-11-18 10:02
閱讀 3454·2021-10-11 10:58
閱讀 3384·2021-09-24 09:47
閱讀 1131·2021-09-22 15:21
閱讀 3963·2021-09-10 11:10
閱讀 3285·2021-09-03 10:28
閱讀 1756·2019-08-30 15:45
閱讀 2150·2019-08-30 14:22