摘要:需要更多地去學習應用。當你調用一個時,它將返回一個迭代器對象。這個迭代器對象擁有一個叫做的方法來幫助你重啟函數并得到下一個值。那么問題來了,咱們也不能手動一直調用方法,你需要一個能夠調用生成器并啟動迭代器的方法。
開始用vue或者react,很多時候我們都會把ES6這個大兄弟加入我們的技術棧中。但是ES6那么多那么多特性,我們需要全部都掌握嗎?秉著二八原則,掌握好常用的,有用的這個可以讓我們快速起飛。
接下來我們就聊聊ES6那些可愛的新特性吧。
1.變量聲明const和let我們都是知道在ES6以前,var關鍵字聲明變量。無論聲明在何處,都會被視為聲明在函數的最頂部(不在函數內即在全局作用域的最頂部)。這就是函數變量提升例如:
function aa() { if(bool) { var test = "hello man" } else { console.log(test) } }
以上的代碼實際上是:
function aa() { var test // 變量提升 if(bool) { test = "hello man" } else { //此處訪問test 值為undefined console.log(test) } //此處訪問test 值為undefined }
所以不用關心bool是否為true or false。實際上,無論如何test都會被創建聲明。
接下來ES6主角登場:
我們通常用let和const來聲明,let表示變量、const表示常量。let和const都是塊級作用域。怎么理解這個塊級作用域?
在一個函數內部
在一個代碼塊內部
說白了 {}大括號內的代碼塊即為let 和 const的作用域。
看以下代碼:
function aa() { if(bool) { let test = "hello man" } else { //test 在此處訪問不到 console.log(test) } }
let的作用域是在它所在當前代碼塊,但不會被提升到當前函數的最頂部。
再來說說const。
const name = "lux" name = "joe" //再次賦值此時會報錯
說一道面試題
var funcs = []; for (var i = 0; i < 10; i++) { funcs.push(function() { console.log(i) }) } funcs.forEach(function(func) { func() })
這樣的面試題是大家常見,很多同學一看就知道輸出 10 十次
但是如果我們想依次輸出0到9呢?兩種解決方法。直接上代碼。
// ES5告訴我們可以利用閉包解決這個問題 var funcs = []; for (var i = 0; i < 10; i++) { func.push( (function(value) { return function() { console.log(value) } }(i)) ) } // es6 for (let i = 0; i < 10; i++) { func.push(function() { console.log(i) }) }
達到相同的效果,es6簡潔的解決方案是不是更讓你心動!!!
2.模板字符串es6模板字符簡直是開發者的福音啊,解決了ES5在字符串功能上的痛點。
第一個用途,基本的字符串格式化。將表達式嵌入字符串中進行拼接。用${}來界定。
//es5 var name = "lux" console.log("hello" + name) //es6 const name = "lux" console.log(`hello ${name}`) //hello lux
第二個用途,在ES5時我們通過反斜杠()來做多行字符串或者字符串一行行拼接。ES6反引號(``)直接搞定。
// es5 var msg = "Hi man! " // es6 const template = `hello world`
對于字符串es6當然也提供了很多厲害的方法。說幾個常用的。
// 1.includes:判斷是否包含然后直接返回布爾值 let str = "hahay" console.log(str.includes("y")) // true // 2.repeat: 獲取字符串重復n次 let s = "he" console.log(s.repeat(3)) // "hehehe" //如果你帶入小數, Math.floor(num) 來處理3.函數
函數默認參數
在ES5我們給函數定義參數默認值是怎么樣?
function action(num) { num = num || 200 //當傳入num時,num為傳入的值 //當沒傳入參數時,num即有了默認值200 return num }
但細心觀察的同學們肯定會發現,num傳入為0的時候就是false, 此時num = 200 與我們的實際要的效果明顯不一樣
ES6為參數提供了默認值。在定義函數時便初始化了這個參數,以便在參數沒有被傳遞進去時使用。
function action(num = 200) { console.log(num) } action() //200 action(300) //300
箭頭函數
ES6很有意思的一部分就是函數的快捷寫法。也就是箭頭函數。
箭頭函數最直觀的三個特點。
不需要function關鍵字來創建函數
省略return關鍵字
繼承當前上下文的 this 關鍵字
//例如:
[1,2,3].map( x => x + 1 )//等同于: [1,2,3].map((function(x){ return x + 1 }).bind(this))
說個小細節。
當你的函數有且僅有一個參數的時候,是可以省略掉括號的。當你函數返回有且僅有一個表達式的時候可以省略{};例如:
var people = name => "hello" + name //參數name就沒有括號
作為參考
var people = (name, age) => { const fullName = "h" + name; return fullName } //如果缺少()或者{}就會報錯4.拓展的對象功能
對象初始化簡寫
ES5我們對于對象都是以鍵值對的形式書寫,是有可能出現鍵值對重名的。例如:
function people(name, age) { return { name: name, age: age }; }
鍵值對重名,ES6可以簡寫如下:
function people(name, age) { return { name, age }; }
ES6 同樣改進了為對象字面量方法賦值的語法。ES5為對象添加方法:
const people = { name: "lux", getName: function() { console.log(this.name) } }
ES6通過省略冒號與 function 關鍵字,將這個語法變得更簡潔
const people = { name: "lux", getName () { console.log(this.name) } }
ES6 對象提供了Object.assign()這個方法來實現淺復制。Object.assign()可以把任意多個源對象自身可枚舉的屬性拷貝給目標對象,然后返回目標對象。第一參數即為目標對象。在實際項目中,我們為了不改變源對象。一般會把目標對象傳為{}
const obj = Object.assign({}, objA, objB)5.更方便的數據訪問--解構
數組和對象是JS中最常用也是最重要表示形式。為了簡化提取信息,ES6新增了解構,這是將一個數據結構分解為更小的部分的過程
ES5我們提取對象中的信息形式如下:
const people = { name: "lux", age: 20 } const name = people.name const age = people.age console.log(name + " --- " + age)
是不是覺得很熟悉,沒錯,在ES6之前我們就是這樣獲取對象信息的,一個一個獲取。現在,解構能讓我們從對象或者數組里取出數據存為變量,例如
//對象 const people = { name: "lux", age: 20 } const { name, age } = people console.log(`${name} --- ${age}`) //數組 const color = ["red", "blue"] const [first, second] = color console.log(first) //"red" console.log(second) //"blue"6.Spread Operator 展開運算符
ES6中另外一個好玩的特性就是Spread Operator 也是三個點兒...接下來就展示一下它的用途。
組裝對象或者數組
//數組 const color = ["red", "yellow"] const colorful = [...color, "green", "pink"] console.log(colorful) //[red, yellow, green, pink] //對象 const alp = { fist: "a", second: "b"} const alphabets = { ...alp, third: "c" } console.log(alphabets) //{ "fist": "a", "second": "b", "third": "c"}
有時候我們想獲取數組或者對象除了前幾項或者除了某幾項的其他項
//數組 const number = [1,2,3,4,5] const [first, ...rest] = number console.log(rest) //2,3,4,5 //對象 const user = { username: "lux", gender: "female", age: 19, address: "peking" } const { username, ...rest } = user console.log(rest) //{"address": "peking", "age": 19, "gender": "female"}
對于 Object 而言,還可以用于組合成新的 Object 。(ES2017 stage-2 proposal) 當然如果有重復的屬性名,右邊覆蓋左邊
const first = { a: 1, b: 2, c: 6, } const second = { c: 3, d: 4 } const total = { ...first, ...second } console.log(total) // { a: 1, b: 2, c: 3, d: 4 }7.import 和 export
import導入模塊、export導出模塊
//全部導入import people from "./example"http://有一種特殊情況,即允許你將整個模塊當作單一對象進行導入//該模塊的所有導出都會作為對象的屬性存在import * as example from "./example.js"console.log(example.name)console.log(example.age)console.log(example.getName())//導入部分import {name, age} from "./example"http:// 導出默認, 有且只有一個默認export default App// 部分導出export class App extend Component {};
以前有人問我,導入的時候有沒有大括號的區別是什么,下面是我在工作中的總結:
1.當用export default people導出時,就用 import people 導入(不帶大括號)
2.一個文件里,有且只能有一個export default。但可以有多個export。
3.當用export name 時,就用import { name }導入(記得帶上大括號)
4.當一個文件里,既有一個export default people, 又有多個export name 或者 export age時,導入就用 import people, { name, age }
5.當一個文件里出現n多個 export 導出很多模塊,導入時除了一個一個導入,也可以用import * as example
在promise之前代碼過多的回調或者嵌套,可讀性差、耦合度高、擴展性低。通過Promise機制,扁平化的代碼機構,大大提高了代碼可讀性;用同步編程的方式來編寫異步代碼,保存線性的代碼邏輯,極大的降低了代碼耦合性而提高了程序的可擴展性。
說白了就是用同步的方式去寫異步代碼。
發起異步請求
fetch("/api/todos") .then(res => res.json()) .then(data => ({ data })) .catch(err => ({ err }));
今天看到一篇關于面試題的很有意思。
setTimeout(function() { console.log(1) }, 0); new Promise(function executor(resolve) { console.log(2); for( var i=0 ; i<10000 ; i++ ) { i == 9999 && resolve(); } console.log(3); }).then(function() { console.log(4); }); console.log(5);
當然以上promise的知識點,這個只是冰山一角。需要更多地去學習應用。
9.Generators生成器( generator)是能返回一個迭代器的函數。生成器函數也是一種函數,最直觀的表現就是比普通的function多了個星號*,在其函數體內可以使用yield關鍵字,有意思的是函數會在每個yield后暫停。
這里生活中有一個比較形象的例子。咱們到銀行辦理業務時候都得向大廳的機器取一張排隊號。你拿到你的排隊號,機器并不會自動為你再出下一張票。也就是說取票機“暫停”住了,直到下一個人再次喚起才會繼續吐票。
OK。說說迭代器。當你調用一個generator時,它將返回一個迭代器對象。這個迭代器對象擁有一個叫做next的方法來幫助你重啟generator函數并得到下一個值。next方法不僅返回值,它返回的對象具有兩個屬性:done和value。value是你獲得的值,done用來表明你的generator是否已經停止提供值。繼續用剛剛取票的例子,每張排隊號就是這里的value,打印票的紙是否用完就這是這里的done。
// 生成器 function *createIterator() { yield 1; yield 2; yield 3; } // 生成器能像正規函數那樣被調用,但會返回一個迭代器 let iterator = createIterator(); console.log(iterator.next().value); // 1 console.log(iterator.next().value); // 2 console.log(iterator.next().value); // 3
那生成器和迭代器又有什么用處呢?
圍繞著生成器的許多興奮點都與異步編程直接相關。異步調用對于我們來說是很困難的事,我們的函數并不會等待異步調用完再執行,你可能會想到用回調函數,(當然還有其他方案比如Promise比如Async/await)。
生成器可以讓我們的代碼進行等待。就不用嵌套的回調函數。使用generator可以確保當異步調用在我們的generator函數運行一下行代碼之前完成時暫停函數的執行。
那么問題來了,咱們也不能手動一直調用next()方法,你需要一個能夠調用生成器并啟動迭代器的方法。就像這樣子的
function run(taskDef) { //taskDef即一個生成器函數 // 創建迭代器,讓它在別處可用 let task = taskDef(); // 啟動任務 let result = task.next(); // 遞歸使用函數來保持對 next() 的調用 function step() { // 如果還有更多要做的 if (!result.done) { result = task.next(); step(); } } // 開始處理過程 step(); }
生成器與迭代器最有趣、最令人激動的方面,或許就是可創建外觀清晰的異步操作代碼。你不必到處使用回調函數,而是可以建立貌似同步的代碼,但實際上卻使用 yield 來等待異步操作結束。
總結
ES6的特性遠不止于此,但對于我們日常的開發開說。這已經是夠夠的了。還有很多有意思的方法。比如findIndex...等等。包括用set來完成面試題常客數組去重問題。我和我的小伙伴們都驚呆了!
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/84178.html
摘要:特意對前端學習資源做一個匯總,方便自己學習查閱參考,和好友們共同進步。 特意對前端學習資源做一個匯總,方便自己學習查閱參考,和好友們共同進步。 本以為自己收藏的站點多,可以很快搞定,沒想到一入匯總深似海。還有很多不足&遺漏的地方,歡迎補充。有錯誤的地方,還請斧正... 托管: welcome to git,歡迎交流,感謝star 有好友反應和斧正,會及時更新,平時業務工作時也會不定期更...
摘要:詳細代碼如下追蹤賦值里面的是子路由設計子路由設計這個比較簡單,每個子路由維護一個路由監聽列表,然后通過調用的函數添加到主路由列表上。 showImg(https://segmentfault.com/img/bVbruD0?w=756&h=378); 前言 鑒于之前使用express和koa的經驗,這兩天想嘗試構建出一個koa精簡版,利用最少的代碼實現koa和koa-router,同時...
摘要:更改文件名之后,在文件中寫入的代碼。控制增加前綴的版本根據頁面中的標簽及標簽中的選擇器屬性,來精簡樣式。 項目地址 既然重新學習了 Gulp,那索性就再把以前用 Gulp 寫的東西拿出來,重新寫一遍。這次寫的時候要把要點記錄下來,不然以后忘了就沒法回憶了。 因為 Gulp 現在使用沒有以前那么多了,所以就不寫復雜的應用了。這次寫一個簡單的 Demo 處理工具,只是為了把 PSD 轉成 ...
摘要:是目前最常用的轉的工具,但即使是,各瀏覽器的支持度也是不一的,因此便產生了本文,找出一些能夠被翻譯成兼容性高代碼的語法。不可用因為會使用到參考資料,用來查瀏覽器兼容性。到底將代碼轉換成什么鳥樣,感謝。官網試驗轉換后的代碼 前言 由于目前各瀏覽器對ES6兼容性較低,再加上需要兼容歷史上各種版本的瀏覽器,因此,使用編譯器將ES6語法轉譯成ES5語法則勢在必行了。babel是目前最常用的ES...
閱讀 3457·2021-11-25 09:43
閱讀 2605·2021-09-22 15:54
閱讀 590·2019-08-30 15:55
閱讀 974·2019-08-30 15:55
閱讀 1998·2019-08-30 15:55
閱讀 1741·2019-08-30 15:53
閱讀 3465·2019-08-30 15:52
閱讀 2039·2019-08-30 12:55