【前端面試:手寫js系列】flat---數(shù)組扁平化

數(shù)組扁平化定義:在前端項目開發(fā)過程中仰坦,偶爾會出現(xiàn)層疊數(shù)據(jù)結(jié)構(gòu)的數(shù)組,需要把多層數(shù)組轉(zhuǎn)換為一級數(shù)組(即提取嵌套數(shù)組元素最終合并為一個數(shù)組)计雌,使其內(nèi)容合并并且展開缎岗。

  • 需求:多維數(shù)組 => 一維數(shù)組
let arr = [1, 2, [3, [4, 5, [6,7]]], 8];  // -> [1, 2, 3, 4, 5, 6, 7, 8]
let str = JSON.stringify(arr); 
  • 方法一:調(diào)用ES6中的flat方法
arr = arr.flat(Infinity);
//不傳參數(shù)時,默認“拉平”一層白粉;傳入一個整數(shù)參數(shù)传泊,整數(shù)即“拉平”的層數(shù);
//Infinity 關(guān)鍵字作為參數(shù)時鸭巴,無論多少層嵌套眷细,都會轉(zhuǎn)為一維數(shù)組
//傳入 <=0 的整數(shù)將返回原數(shù)組,不“拉平”
//如果原數(shù)組有空位鹃祖,flat()方法會跳過空位
  • 手寫一個數(shù)組扁平化flat方法
    思路:
    1溪椎、遍歷數(shù)組的每一個元素;
    2恬口、判斷元素是否為數(shù)組校读;
    3、將是數(shù)組的元素進行展開祖能。

遍歷數(shù)組的方案:
for循環(huán)
for...of
for...in
forEach()
entries()
keys()
values()
reduce()
map()

const arr = [1, 2, 3, 4, [1, 2, 3, [1, 2, 3, [1, 2, 3]]], 5, "string", { name: "小馬同學" }];
//for循環(huán)遍歷
for (var i = 0; i < arr.length; i++){
    console.log(arr[i]);
}
//for...of遍歷
for (let item of arr) {
    console.log(item)
}
//for...in遍歷
for (let item in arr) {
    console.log(arr[item]);
}
//forEach遍歷
arr.forEach(value = >{
    console.log(value);
});
// entries()遍歷
for (let [index, value] of arr.entries()) {
  console.log(value);
}
// keys() 遍歷
for (let index of arr.keys()) {
  console.log(arr[index]);
}
// values() 遍歷
for (let value of arr.values()) {
  console.log(value);
}
//map() 遍歷
arr.map(value = > {
    console.log(value);
}

判斷元素是否是數(shù)組的方案:
instanceof
constructor
object.prototype.toString.call
isArray

const arr = [1, 2, 3, 4, [1, 2, 3, [1, 2, 3, [1, 2, 3]]], 5, "string", { name: "小馬同學" }];
arr instanceof Array; // true
arr.constructor === Array; //true
object.prototype.toString.call(arr) === '[object Array]'; //true
Array.isArray(arr); //true

instanceof 操作符是假定只有一種全局環(huán)境歉秫,如果網(wǎng)頁中包含多個框架,多個全局環(huán)境养铸,如果你從一個框架向另一個框架傳入一個數(shù)組雁芙,那么傳入的數(shù)組與在第二個框架中原生創(chuàng)建的數(shù)組分別具有各自不同的構(gòu)造函數(shù)。(所以在這種情況下會不準確)
typeof 操作符對數(shù)組取類型將返回 object
因為 constructor 可以被重寫钞螟,所以不能確保一定是數(shù)組兔甘。

//constructor重寫,判斷字符串為數(shù)組
const str = 'abc';
str.constructor = Array;
str.constructor === Array 
// true

將數(shù)組元素進行展開一層的方案:
擴展運算法 + concat(concat() 方法用于合并兩個或多個數(shù)組,在拼接的過程中加上擴展運算符會展開一層數(shù)組)
concat + apply(主要是利用 apply 在綁定作用域時,傳入的第二個參數(shù)是一個數(shù)組或者類數(shù)組對象蹋半,其中的數(shù)組元素將作為單獨的參數(shù)傳給 func 函數(shù)。也就是在調(diào)用 apply 函數(shù)的過程中澡匪,會將傳入的數(shù)組一個一個的傳入到要執(zhí)行的函數(shù)中,也就是相當對數(shù)組進行了一層的展開提岔。)
toString + split(不推薦使用 toString + split 方法仙蛉,因為操作字符串是很危險的事情,如果數(shù)組中的元素所有都是數(shù)字的話碱蒙,toString + split 是可行的荠瘪,并且是一步搞定夯巷。)

const arr = [1, 2, 3, 4, [1, 2, 3, [1, 2, 3, [1, 2, 3]]], 5, "string", { name: "小馬同學" }];
//擴展運算法 + concat
[ ].concat(...arr);
// concat + apply
[ ].concat.apply([ ], arr);
//以上兩種方法,輸出結(jié)果:
//[1, 2, 3, 4, 1, 2, 3, [1, 2, 3, [1, 2, 3]], 5, "string", { name: "小馬同學" }];
//toString + split
const arr2 = [1, 2, 3, 4, [1, 2, 3, [1, 2, 3, [1, 2, 3]]]];
arr2.toString().split(',').map(v = > parseInt(v));
//[1, 2, 3, 4, 1, 2, 3, 1, 2, 3, 1, 2, 3]

最終手寫的flat()方法為:

const arr = [1, 2, 3, 4, 5, [1, 2, [1, 2, 3, [4, 5]]], 6, 'flat方法', {name: '小馬同學' }];
function flat(arr){
    let arrResult = [];
    for(let value of arr){
          if(Array.isArray(value)){
                arrResult.push(...arguments.callee(value));  //遞歸
          }else{
                arrResult.push(value);
          }
    }
return arrResult;
}
console.log(flat(arr));
//[1, 2, 3, 4, 5, 1, 2, 1, 2, 3, 4, 5, 6, 'flat方法', {name: '小馬同學' }]

arguments.callee(value) = flat(value)趁餐,為了解決函數(shù)的執(zhí)行與函數(shù)名flat緊密耦合的情況。
擴展運算符是三個點(...)臀突,比較像 rest 參數(shù)的逆運算候学,作用是將一個數(shù)組轉(zhuǎn)為用逗號分隔的參數(shù)序列纵散。

  • 方法二:repalce + split
arr = str.replace(/(\[|\]))/g, '').split(',');
  • 方法三:replace + JSON.parse
str = str.replace(/(\[|\]))/g, '');
str = '[' + str +']';
arr = JSON.parse(str);
  • 方法四:普通遞歸
let result = [];
let fn = function(arr){
    for(let i = 0; i < arr.length; i++){
        let item = arr[i]; //重點
        if(Array.isArray(arr[i])){
            fn(item);  //遞歸
        }else{
            result.push(item);
        }
   }
}
  • 方法五:利用reduce函數(shù)迭代
const arr = [1, 2, 3, 4, 5, [1, 2, [1, 2, 3, [4, 5]]], 6, 'flat方法', {name: '小馬同學' }];
function flatten(arr){
    return arr.reduce((pre,cur) =>{
        return pre.concat(Array.isArray(cur) ? flatten(cur) : cur);
    }, []);
}
console.log(flatten(arr)); 
//[1, 2, 3, 4, 5, 1, 2, 1, 2, 3, 4, 5, 6, 'flat方法', {name: '小馬同學' }]
  • 方法六:擴展運算符
let arr = [1, 2, 3, 4, [5, 6, [7, 8]]]; //這個方法數(shù)組中的元素只能是數(shù)字伍掀?
while (arr.some((item) => { return Array.isArray(item) })) {
arr = [].concat(…arr);
}
console.log(arr)

參考博客如下,注明一下出處濒蒋,感謝大神們甸各,希望自己可以多多練習,多多回顧:
三元博客
JS數(shù)組reduce()方法詳解及高級技巧
數(shù)組flat方法實現(xiàn)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末儒恋,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子诫尽,更是在濱河造成了極大的恐慌,老刑警劉巖减途,帶你破解...
    沈念sama閱讀 206,602評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件税产,死亡現(xiàn)場離奇詭異辟拷,居然都是意外死亡阐斜,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評論 2 382
  • 文/潘曉璐 我一進店門莉测,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人忍抽,你說我怎么就攤上這事鸠项。” “怎么了?”我有些...
    開封第一講書人閱讀 152,878評論 0 344
  • 文/不壞的土叔 我叫張陵泽腮,是天一觀的道長。 經(jīng)常有香客問我衣赶,道長诊赊,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,306評論 1 279
  • 正文 為了忘掉前任府瞄,我火速辦了婚禮碧磅,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘遵馆。我一直安慰自己鲸郊,他們只是感情好,可當我...
    茶點故事閱讀 64,330評論 5 373
  • 文/花漫 我一把揭開白布货邓。 她就那樣靜靜地躺著秆撮,像睡著了一般。 火紅的嫁衣襯著肌膚如雪换况。 梳的紋絲不亂的頭發(fā)上职辨,一...
    開封第一講書人閱讀 49,071評論 1 285
  • 那天,我揣著相機與錄音戈二,去河邊找鬼舒裤。 笑死,一個胖子當著我的面吹牛觉吭,可吹牛的內(nèi)容都是我干的腾供。 我是一名探鬼主播,決...
    沈念sama閱讀 38,382評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼鲜滩,長吁一口氣:“原來是場噩夢啊……” “哼伴鳖!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起徙硅,我...
    開封第一講書人閱讀 37,006評論 0 259
  • 序言:老撾萬榮一對情侶失蹤榜聂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后闷游,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體峻汉,經(jīng)...
    沈念sama閱讀 43,512評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,965評論 2 325
  • 正文 我和宋清朗相戀三年脐往,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片扳埂。...
    茶點故事閱讀 38,094評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡业簿,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出阳懂,到底是詐尸還是另有隱情梅尤,我是刑警寧澤柜思,帶...
    沈念sama閱讀 33,732評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站巷燥,受9級特大地震影響赡盘,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜缰揪,卻給世界環(huán)境...
    茶點故事閱讀 39,283評論 3 307
  • 文/蒙蒙 一陨享、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧钝腺,春花似錦抛姑、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,286評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至毫目,卻和暖如春蔬啡,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背镀虐。 一陣腳步聲響...
    開封第一講書人閱讀 31,512評論 1 262
  • 我被黑心中介騙來泰國打工星爪, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人粉私。 一個月前我還...
    沈念sama閱讀 45,536評論 2 354
  • 正文 我出身青樓顽腾,卻偏偏與公主長得像,于是被迫代替她去往敵國和親诺核。 傳聞我的和親對象是個殘疾皇子抄肖,可洞房花燭夜當晚...
    茶點故事閱讀 42,828評論 2 345

推薦閱讀更多精彩內(nèi)容