自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

精學(xué)手撕系列——數(shù)組扁平化

開發(fā)
在前端面試中,手寫flat是非?;A(chǔ)的面試題,通常出現(xiàn)在筆試或者第一輪面試中,主要考察面試者基本的手寫代碼能力和JavaScript的基本功。

 今天就帶大家從0了解flat特性到手寫實(shí)現(xiàn)flat,再到接住面試官的連環(huán)追問中重新學(xué)習(xí)一遍數(shù)組扁平化flat方法

[[342383]]

Array.prototype.flat()
一段代碼總結(jié)Array.prototype.flat()特性

注:數(shù)組拍平方法 Array.prototype.flat() 也叫數(shù)組扁平化、數(shù)組拉平、數(shù)組降維。

  1. let arr = [12, 23, [34, 56, [78, 90, 100, [110, 120, 130]]]]; 
  2.  
  3. console.log(arr.flat());  
  4. // [ 12, 23, 34, 56, [ 78, 90, 100, [ 110, 120, 130 ] ] ] 
  5.  
  6. console.log(arr.flat(2)); 
  7. // [ 12, 23, 34, 56, 78, 90, 100, [ 110, 120, 130 ] ] 
  8.  
  9. console.log(arr.flat(Infinity)); 
  10. // [12, 23, 34, 56, 78, 90, 100, 110, 120, 130] 
  11.  
  12. console.log(arr.flat(0)); 
  13. // [12, 23, [34, 56, [78, 90, 100, [110, 120, 130]]]]; 
  14.  
  15. console.log(arr.flat(-10)); 
  16. // [12, 23, [34, 56, [78, 90, 100, [110, 120, 130]]]]; 
  17.  
  18. let arr2 = [12, 23, [34, 56, ,]] 
  19. console.log(arr.flat()); 
  20. // [ 12, 23, 34, 56 ] 

Array.prototype.flat() 特性總結(jié)

Array.prototype.flat() 用于將嵌套的數(shù)組“拉平”,變成一維的數(shù)組。該方法返回一個(gè)新數(shù)組,對(duì)原數(shù)據(jù)沒有影響。

不傳參數(shù)時(shí),默認(rèn)“拉平”一層,可以傳入一個(gè)整數(shù),表示想要“拉平”的層數(shù)。

傳入 <=0 的整數(shù)將返回原數(shù)組,不“拉平”

Infinity 關(guān)鍵字作為參數(shù)時(shí),無論多少層嵌套,都會(huì)轉(zhuǎn)為一維數(shù)組

如果原數(shù)組有空位,Array.prototype.flat() 會(huì)跳過空位。

面試官 N 連問:
第一問:下面數(shù)組如何實(shí)現(xiàn)扁平化?

  1. let arr = [ 
  2.   [1, 2, 2], 
  3.   [3, 4, 5, 5], 
  4.   [6, 7, 8, 9, [11, 12, [12, 13, [14]]]], 10 
  5. ]; 

小伙伴首先想到的肯定是用 ES6 的Array.prototype.flat方法呀

方法一:flat

  1. arr = arr.flat(2); 
  2. // [ 1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, [ 12, 13, [ 14 ] ], 10 ] 

當(dāng)flat中傳入數(shù)字時(shí),是扁平對(duì)應(yīng)的層數(shù),顯然這不是我們想要的,因?yàn)樗€沒有完全展開。

這是,flat函數(shù)中就為我們提供了一個(gè)參數(shù)Infinity,譯為無窮的意思。

  1. arr = arr.flat(Infinity); 
  2. /* [ 
  3.    1,  2,  2, 3,  4,  5,  5, 
  4.    6,  7,  8, 9, 11, 12, 12, 
  5.   13, 14, 10 
  6. ] */ 

當(dāng)我們不知道數(shù)組中嵌套了幾維數(shù)組時(shí),我們可以用Infinity這個(gè)參數(shù),幫我們?nèi)空归_。

第二問:還有其它的辦法嗎?因?yàn)樗鼈冊(cè)诟甙姹緸g覽器并不兼容
方法二:轉(zhuǎn)換為字符串,再把字符串對(duì)象用,轉(zhuǎn)換成數(shù)組
可以先把多維數(shù)組先轉(zhuǎn)換為字符串,再基于,分隔符將字符串對(duì)象分割成字符串?dāng)?shù)組

toString() 扁平化數(shù)組

  1. arr = arr.toString(); 
  2. // "1,2,2,3,4,5,5,6,7,8,9,11,12,12,13,14,10" 
  3.  
  4. arr = arr.toString().split(','); 
  5. // ["1""2""2""3""4""5""5""6""7""8""9""11""12""12""13""14""10"
  6.  
  7. arr = arr.toString().split(',').map(item => parseFloat(item)); 
  8. // [1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, 12, 13, 14, 10] 

除了上面的方法還有什么方法轉(zhuǎn)換為字符串呢?

JSON.stringify()扁平化數(shù)組

  1. arr = JSON.stringify(arr); 
  2. // "[[1,2,2],[3,4,5,5],[6,7,8,9,[11,12,[12,13,[14]]]],10]" 
  3.  
  4. arr = JSON.stringify(arr).replace(/(\[|\])/g, ''); 
  5. // "1,2,2,3,4,5,5,6,7,8,9,11,12,12,13,14,10" 
  6.  
  7. arr = JSON.stringify(arr).replace(/(\[|\])/g, '').split(',').map(item=>parseFloat(item)); 
  8. // [1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, 12, 13, 14, 10] 

方法三:循環(huán)驗(yàn)證是否為數(shù)組
基于數(shù)組的some方法,只要數(shù)組里面有一項(xiàng)元素是數(shù)組就繼續(xù)循環(huán),扁平數(shù)組

核心:[].concat(...arr)

  1. whilte (arr.some(item => Array.isArray(item))) { 
  2.  arr = [].concat(...arr); 
  3.  
  4. console.log(arr); // [1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, 12, 13, 14, 10] 

第三問:能自己實(shí)現(xiàn)一個(gè) flat 扁平化嗎?
先回答:能??!再跟著我分析思路:

如何實(shí)現(xiàn)呢,其實(shí)思路非常簡(jiǎn)單:在數(shù)組中找到是數(shù)組類型的元素,然后將他們展開,這就是flat方法的關(guān)鍵思路

實(shí)現(xiàn)思路:

循環(huán)數(shù)組里的每一個(gè)元素
判斷該元素是否為數(shù)組
是數(shù)組的話,繼續(xù)循環(huán)遍歷這個(gè)元素——數(shù)組
不是數(shù)組的話,把元素添加到新的數(shù)組中
實(shí)現(xiàn)流程:

創(chuàng)建一個(gè)空數(shù)組,用來保存遍歷到的非數(shù)組元素
創(chuàng)建一個(gè)循環(huán)遍歷數(shù)組的函數(shù),cycleArray
取得數(shù)組中的每一項(xiàng),驗(yàn)證Array.isArray()
數(shù)組的話,繼續(xù)循環(huán)
非數(shù)組的話,添加到新數(shù)組中
返回新數(shù)組對(duì)象
ES5 實(shí)現(xiàn) flat 扁平化方法

  1. let arr = [ 
  2.     [1, 2, 2], 
  3.     [3, 4, 5, 5], 
  4.     [6, 7, 8, 9, [11, 12, [12, 13, [14]]]], 10 
  5. ]; 
  6.  
  7. function myFlat() { 
  8.   _this = this; // 保存 this:arr 
  9.   let newArr = []; 
  10.   // 循環(huán)arr中的每一項(xiàng),把不是數(shù)組的元素存儲(chǔ)到 newArr中 
  11.   let cycleArray = (arr) => { 
  12.     for (let i=0; i< arr.length; i++) { 
  13.       let item = arr[i]; 
  14.       if (Array.isArray(item)) { // 元素是數(shù)組的話,繼續(xù)循環(huán)遍歷該數(shù)組 
  15.         cycleArray(item); 
  16.         continue
  17.       } else
  18.         newArr.push(item); // 不是數(shù)組的話,直接添加到新數(shù)組中 
  19.       } 
  20.     } 
  21.   } 
  22.   cycleArray(_this); // 循環(huán)數(shù)組里的每個(gè)元素 
  23.   return newArr; // 返回新的數(shù)組對(duì)象 
  24.  
  25. Array.prototype.myFlat = myFlat; 
  26.  
  27. arr = arr.myFlat(); // [1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, 12, 13, 14, 10] 

ES6 實(shí)現(xiàn) flat 扁平化方法

  1. const myFlat = (arr) => { 
  2.   let newArr = []; 
  3.   let cycleArray = (arr) => { 
  4.     for(let i = 0; i < arr.length; i++) { 
  5.       let item = arr[i]; 
  6.       if (Array.isArray(item)) { 
  7.         cycleArray(item); 
  8.         continue
  9.       } else { 
  10.         newArr.push(item); 
  11.       } 
  12.     } 
  13.   } 
  14.   cycleArray(arr); 
  15.   return newArr; 
  16.  
  17. myFlat(arr); // [1, 2, 2, 3, 4, 5, 5, 6, 7, 8, 9, 11, 12, 12, 13, 14, 10] 

第四問:請(qǐng)使用reduce實(shí)現(xiàn)flat函數(shù)
相信很多面試官都會(huì)指定讓面試者用reduce方法實(shí)現(xiàn)flat函數(shù)

其實(shí)思路也是一樣的,在實(shí)現(xiàn)之前,先來看一下

它的核心:[].concat(...arr)

但是它只能將數(shù)組元素展開一層,來看下面例子:

  1. let arr2 = [12, 23, [34, 56, [78, 90, 100]]]; 
  2. [].concat(...arr2); 
  3. // [ 12, 23, 34, 56, [ 78, 90, 100 ] ] 

細(xì)心的同學(xué)可以發(fā)現(xiàn)[].concat(...arr)只能展開一層數(shù)組元素,當(dāng)有更深層次的,是無法展開的

接下來,我們來看看用reduce怎么實(shí)現(xiàn)?

  1. let arr = [12, 23, [34, 56, [78, 90, 100, [110, 120, 130, 140]]]]; 
  2. const myFlat = arr => { 
  3.   return arr.reduce((pre, cur) => { 
  4.     return pre.concat(cur); 
  5.   }, []); 
  6. }; 
  7. console.log(myFlat(arr)); 
  8. // [ 12, 23, 34, 56, [ 78, 90, 100, [ 110, 120, 130, 140 ] ] ] 
  9.  
  10. const myFlat = arr => { 
  11.   return arr.reduce((pre, cur) => { 
  12.     return pre.concat(Array.isArray(cur) ? myFlat(cur) : cur); 
  13.   }, []); 
  14. }; 
  15. console.log(myFlat(arr)); 
  16. // [12, 23, 34, 56, 78, 90, 100, 110, 120, 130, 140] 

上面代碼中的Array.isArray(cur)和myFlat(cur)實(shí)際就好比與遍歷數(shù)組每一項(xiàng),看它是不是數(shù)組元素,

如果是的話,則繼續(xù)遞歸遍歷,不是的話直接數(shù)組合并非數(shù)組元素。

第五問:使用棧的思想實(shí)現(xiàn)flat函數(shù)
棧思想: 后進(jìn)先出的數(shù)據(jù)結(jié)構(gòu)

實(shí)現(xiàn)思路:

不斷獲取并刪除棧中最后一個(gè)元素A,判斷A是否為數(shù)組元素,直到棧內(nèi)元素為空,全部添加到newArr

是數(shù)組,則push到棧中,繼續(xù)循環(huán)棧內(nèi)元素,直到棧為空
不是數(shù)組,則unshift添加到newArr中

  1. // 棧思想 
  2. function flat(arr) { 
  3.   const newArr = []; 
  4.   const stack = [].concat(arr);  // 將數(shù)組元素拷貝至棧,直接賦值會(huì)改變?cè)瓟?shù)組 
  5.   //如果棧不為空,則循環(huán)遍歷 
  6.   while (stack.length !== 0) { 
  7.     const val = stack.pop(); // 刪除數(shù)組最后一個(gè)元素,并獲取它 
  8.     if (Array.isArray(val)) { 
  9.       stack.push(...val); // 如果是數(shù)組再次入棧,并且展開了一層 
  10.     } else { 
  11.       newArr.unshift(val); // 如果不是數(shù)組就將其取出來放入結(jié)果數(shù)組中 
  12.     } 
  13.   } 
  14.   return newArr; 
  15.  
  16. let arr = [12, 23, [34, 56, [78, 90, 100, [110, 120, 130, 140]]]]; 
  17. console.log(flat(arr)); 
  18. // [12, 23, 34, 56, 78, 90, 100, 110, 120, 130, 140] 

本文總結(jié)
看完這篇文章的同學(xué),可以在面試的時(shí)候分類,分思想給面試官描述,可以先說我用哪幾種思想實(shí)現(xiàn)過,它們的寫法又分別有什么不同。

 

責(zé)任編輯:姜華 來源: 前端時(shí)光屋
相關(guān)推薦

2020-09-17 14:04:32

拷貝

2013-09-22 15:43:27

扁平化UI設(shè)計(jì)

2013-09-22 15:03:27

扁平化UI設(shè)計(jì)

2013-09-22 14:47:50

扁平化UI設(shè)計(jì)

2013-09-22 14:56:26

扁平化UI設(shè)計(jì)

2013-09-22 14:30:43

扁平化設(shè)計(jì)

2013-09-22 16:22:58

扁平化UI設(shè)計(jì)

2013-09-22 16:36:07

扁平化UI設(shè)計(jì)

2013-09-22 16:17:59

扁平化UI設(shè)計(jì)

2021-11-07 14:37:59

列表編碼扁平化

2013-06-09 11:04:07

設(shè)計(jì)扁平化設(shè)計(jì)平面化設(shè)計(jì)

2013-09-25 10:52:27

Android 4.4扁平化

2013-09-22 16:30:13

扁平化UI設(shè)計(jì)

2013-09-23 10:16:04

扁平化UI設(shè)計(jì)

2019-04-08 16:29:28

多云扁平化云計(jì)算

2013-07-11 09:45:48

扁平化扁平化設(shè)計(jì)

2013-01-28 09:39:05

用戶界面扁平化設(shè)計(jì)

2021-11-24 08:43:02

扁平化函數(shù)數(shù)組

2013-09-22 15:09:04

扁平化UI設(shè)計(jì)

2012-02-20 14:48:30

網(wǎng)絡(luò)扁平化
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)