《前端面試題:JS數組去重》

JavaScript數組去重終極指南:從基礎到高級的多種方法(附面試題解析)

在前端開發中,數組去重是JavaScript中最常見的需求之一。本文將全面解析8種數組去重方法,包括基礎實現、ES6新特性、性能優化等,并附上面試常見問題解析。


一、基礎去重方法(ES5及以前)
  1. 雙重循環法
    最基礎的去重方法,通過兩層循環比較元素:

    function unique(arr) {const result = [];for (let i = 0; i < arr.length; i++) {let isDuplicate = false;for (let j = 0; j < result.length; j++) {if (arr[i] === result[j]) {isDuplicate = true;break;}}if (!isDuplicate) result.push(arr[i]);}return result;
    }// 示例
    console.log(unique([1, 2, 2, 3])); // [1, 2, 3]
    

    時間復雜度:O(n2)
    適用場景:小型數組,兼容性要求高

  2. indexOf優化法
    使用indexOf替代內層循環:

    function unique(arr) {const result = [];for (let i = 0; i < arr.length; i++) {if (result.indexOf(arr[i]) === -1) {result.push(arr[i]);}}return result;
    }
    

    注意indexOf內部也是循環,本質上還是O(n2)復雜度

  3. 排序相鄰比較法
    先排序后比較相鄰元素:

    function unique(arr) {arr.sort();const result = [arr[0]];for (let i = 1; i < arr.length; i++) {if (arr[i] !== arr[i-1]) {result.push(arr[i]);}}return result;
    }
    

    缺點:會改變原數組順序,不適用于對象類型


二、ES6+ 高效去重方法
  1. Set數據結構法(最常用)
    ES6的Set自動處理唯一值:

    const unique = arr => [...new Set(arr)];// 示例
    console.log(unique([1, 2, 2, '2', NaN, NaN])); 
    // [1, 2, "2", NaN]
    

    特點

    • 代碼最簡潔(9個字符最短實現)
    • 支持NaN去重(NaN === NaN為false,但Set能識別)
    • 時間復雜度和空間復雜度均為O(n)
  2. Map數據結構法
    利用Map的鍵唯一性:

    function unique(arr) {const map = new Map();return arr.filter(item => !map.has(item) && map.set(item, true));
    }
    

    優勢:保留原始順序,適合需要順序的場景

  3. filter + indexOf
    利用索引位置判斷:

    const unique = arr => arr.filter((item, index) => arr.indexOf(item) === index);
    

    缺點indexOf無法識別NaN,時間復雜度O(n2)

  4. reduce累積法
    使用reduce實現優雅去重:

    const unique = arr => arr.reduce((acc, cur) => acc.includes(cur) ? acc : [...acc, cur], []
    );
    

三、特殊場景處理
  1. 對象數組去重
    根據對象屬性去重:

    function uniqueByKey(arr, key) {const map = new Map();return arr.filter(item => {const identifier = item[key];return !map.has(identifier) && map.set(identifier, true);});
    }// 示例
    const users = [{id: 1, name: 'Alice'},{id: 1, name: 'Alice'}, {id: 2, name: 'Bob'}
    ];
    console.log(uniqueByKey(users, 'id')); 
    // 兩個Alice對象只會保留一個
    
  2. 多維數組去重
    遞歸處理嵌套數組:

    function deepUnique(arr) {const flatArr = arr.flat(Infinity);return [...new Set(flatArr)];
    }// 示例
    console.log(deepUnique([1, [2, [3, 2]], 4])); // [1, 2, 3, 4]
    
  3. 混合類型精確去重
    區分數字1和字符串’1’:

    function strictUnique(arr) {const seen = new Map();return arr.filter(item => {const type = typeof item;const key = `${type}-${item}`;return !seen.has(key) && seen.set(key, true);});
    }// 示例
    console.log(strictUnique([1, '1', 2, 2])); // [1, "1", 2]
    

四、面試高頻問題與陷阱解析
  1. 問題:Set去重有什么缺陷?

    • 不區分-0+0(Set認為它們相等)
    • 對象內容相同但引用不同時無法去重
    const obj = {a: 1};
    console.log(unique([obj, obj, {a: 1}])); 
    // [obj, {a:1}] → 兩個不同對象
    
  2. 問題:如何實現O(n)時間復雜度的去重?
    :使用Set或Map數據結構,它們的查找操作是O(1)復雜度

  3. 陷阱:indexOf無法識別NaN

    [NaN].indexOf(NaN) // -1 → 無法識別
    [...new Set([NaN, NaN])] // [NaN] → 正確識別
    
  4. 問題:如何保留去重后的原始順序?
    :使用Map或對象記錄首次出現位置:

    function orderedUnique(arr) {return [...new Map(arr.map(item => [item, item])).values()];
    }
    
  5. 性能大比拼(10,000個元素測試):

    方法耗時(ms)可讀性適用場景
    雙重循環1200★★☆小型數組
    Set2.5★★★★★現代瀏覽器
    filter + indexOf850★★★☆簡單去重
    Map3.0★★★★☆需保留順序的場景

五、去重方法總結表
方法代碼復雜度時間復雜度特殊類型支持適用場景
雙重循環O(n2)所有類型兼容性要求高的老項目
Set極低O(n)? NaN現代項目首選
MapO(n)? 對象引用需要保留順序的場景
filter + indexOfO(n2)? NaN簡單數組去重
reduceO(n2)? NaN函數式編程場景
排序相鄰比較O(n log n)? 混合類型純數字/字符串數組
對象屬性去重O(n)? 按指定屬性去重對象數組去重

六、最佳實踐與使用建議
  1. 現代項目首選

    // 99%場景使用Set足夠
    const unique = arr => [...new Set(arr)];
    
  2. 需要兼容IE時

    // 使用對象+類型標記的polyfill
    function legacyUnique(arr) {const seen = {};return arr.filter(item => {const key = typeof item + JSON.stringify(item);return seen.hasOwnProperty(key) ? false : (seen[key] = true);});
    }
    
  3. 大數組優化
    對于超過10,000個元素的數組:

    function bigArrayUnique(arr) {const set = new Set();const result = [];for (let i = 0; i < arr.length; i++) {if (!set.has(arr[i])) {set.add(arr[i]);result.push(arr[i]);}}return result;
    }
    
  4. TypeScript強化版

    function typedUnique<T>(arr: T[]): T[] {return Array.from(new Set(arr));
    }// 對象數組按key去重
    function uniqueByKey<T>(arr: T[], key: keyof T): T[] {const map = new Map<any, T>();arr.forEach(item => {const identifier = item[key];if (!map.has(identifier)) {map.set(identifier, item);}});return Array.from(map.values());
    }
    

掌握這些去重方法,不僅能輕松應對面試,更能根據實際場景選擇最優方案。記住:當簡單方案(如Set)能滿足需求時,不要過度設計。建議在項目中統一封裝去重工具函數。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/85172.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/85172.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/85172.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

基于51單片機的智能小車:按鍵調速、障礙跟蹤、紅外循跡與數碼管顯示(一個合格的單片機課設)

引言 在嵌入式系統領域&#xff0c;51單片機因其簡單易用、成本低廉的特點&#xff0c;一直是入門學習的理想平臺。今天我將分享一個基于51單片機的多功能智能小車項目&#xff0c;它集成了按鍵PWM調速、障礙物跟蹤、紅外循跡和數碼管顯示四大功能。這個項目不僅涵蓋了嵌入式開…

Java異常處理(try-catch-finally):像醫生一樣處理程序的“感冒”

&#x1f525;「炎碼工坊」技術彈藥已裝填&#xff01; 點擊關注 → 解鎖工業級干貨【工具實測|項目避坑|源碼燃燒指南】 一、從一個真實問題開始&#xff1a;為什么需要異常處理&#xff1f; 假設你正在開發一個文件讀取工具&#xff0c;用戶輸入文件名后&#xff0c;程序會讀…

PostgreSQL 數據庫故障與性能高效實時監測技術深度解析

關鍵詞&#xff1a; postgresql 故障與性能監控 &#x1f4d1; 文章目錄 1. 引言與監控重要性 2. PostgreSQL監控體系架構 3. 故障監控核心技術 4. 性能監控關鍵指標 5. 實時監測技術實現 6. 監控工具選型與部署 7. 故障預警與自動化響應 8. 性能調優監控策略 9. 最佳…

logrotate 踩坑

我的logrotate配置&#xff0c;原本運行正常&#xff0c;最近幾天發現輪轉失敗&#xff0c;兩個目錄下的日志全部無法輪轉&#xff0c;于是開始排查問題 /data01/logs/test1/*.log /data01/logs/test2/*.log {missingokrotate 1notifemptycreate 0644 www-data admsharedscrip…

FastGPT、百度智能體、Coze與MaxKB四大智能體平臺在政務場景下的深度對比

在生成式AI技術快速迭代的浪潮中&#xff0c;百度智能體平臺、Coze、FastGPT和MaxKB作為四大智能體開發平臺&#xff0c;憑借差異化的技術路徑和功能特性&#xff0c;正在重塑政務AI應用的開發范式。本文從功能實現、政務場景適應性等維度展開深度解析&#xff0c;為開發者提供…

基于SpringBoot的美食分享平臺-038

一、項目技術棧 Java開發工具&#xff1a;JDK1.8 后端框架&#xff1a;SpringBoot 前端&#xff1a;采用HTML和Vue相結合開發 數據庫&#xff1a;MySQL5.7和Navicat管理工具結合 服務器&#xff1a;Tomcat8.5 開發軟件&#xff1a;IDEA / Eclipse 是否Maven項目&#xff1a;是 …

【C++第三方包安裝】Windows與Linux安裝配置redis-plus-plus指南

前言 下面主要是對于兩種環境安裝、配置、使用C的第三方包&#xff08;redis&#xff09;&#xff0c;對于其他的第三方庫&#xff0c;也可以使用類似的方法進行類比安裝。 且大多數的第三方庫都可以利用工具一鍵安裝或手動編譯安裝。 Windows 要在Windows系統上快速安裝和使…

springboot入門之路(二)

系列文章目錄 springboot入門之路&#xff08;一&#xff09;連續的學習漸進之路。閱讀點擊&#xff1a;springboot入門之路(一) 文章目錄 系列文章目錄3.springboot配置及注意事項3.1繼承starter parent3.2使用沒有父POM的Spring Boot3.3配置java的編譯的版本3.4使用"de…

【開源解析】基于Python+Qt打造智能應用時長統計工具 - 你的數字生活分析師

&#x1f4ca; 【開源解析】基于PythonQt打造智能應用時長統計工具 - 你的數字生活分析師 &#x1f308; 個人主頁&#xff1a;創客白澤 - CSDN博客 &#x1f525; 系列專欄&#xff1a;&#x1f40d;《Python開源項目實戰》 &#x1f4a1; 熱愛不止于代碼&#xff0c;熱情源自…

PHP語法基礎篇(三):類型轉換與常量

"在完成PHP輸出函數和字符串操作的學習后&#xff0c;本篇筆記將記錄 類型轉換和 常量應用的學習過程。作為語法基礎篇的第三部分&#xff0c;將重點關注&#xff1a; 類型轉換數學函數常量定義&#xff1a;define() 與const 的使用差異魔術常量應用&#xff1a;__LINE__ …

Linux lsof 命令詳解+實例

&#x1f468;?&#x1f393;博主簡介 &#x1f3c5;CSDN博客專家 ??&#x1f3c5;云計算領域優質創作者 ??&#x1f3c5;華為云開發者社區專家博主 ??&#x1f3c5;阿里云開發者社區專家博主 &#x1f48a;交流社區&#xff1a;運維交流社區 歡迎大家的加入&#xff01…

【Cobalt Strike手冊】客戶端界面功能

工具欄 頂部的工具欄提供了快速訪問的功能&#xff0c;這些圖片的功能從左到右功能以此如下表 創建新的連接斷開當前的TeamServerListeners監聽器列表以圖形化展示表格形式展示表格展示目標管理Web服務查看獲取到的認證信息查看下載的文件查看鍵盤記錄查看截屏記錄 圖形化會話…

FastAPI本地文檔的定制技巧

磨刀不誤砍柴工&#xff0c;一份清晰的API文檔能讓前后端協作效率翻倍——源滾滾如是說 在前后端分離開發的今天&#xff0c;接口文檔的質量直接決定了團隊協作的效率。作為Python領域最受矚目的現代Web框架&#xff0c;FastAPI最大的亮點之一是其自動化交互式文檔功能。但很多…

Python 標準庫概覽

Python 標準庫非常龐大,所提供的組件涉及范圍十分廣泛,使用標準庫我們可以讓您輕松地完成各種任務。 以下是一些 Python3 標準庫中的模塊: os 模塊:os 模塊提供了許多與操作系統交互的函數,例如創建、移動和刪除文件和目錄,以及訪問環境變量等。 sys 模塊:sys 模塊提供…

AI大模型:(二)4.1 文生圖(Text-to-Image)模型發展史

目錄 1.介紹 2.發展歷史 2.1.早期探索階段(1980-2014 年) 2.1.1.卷積神經網絡(CNN) 2.1.2.生成對抗網絡(GAN)的提出 2.2.GAN主導時代(2015-2018 年) 2.2.1.高分辨率GAN的突破 2.2.2.文本-圖像對齊的改進 2.3. Diffusion革命(2021–2022) 2.3.1.擴散模型(D…

vue3實現輪播渲染多張圖每張進行放大縮小拖拽功能互不影響

vue3實現輪播渲染多張圖每張進行放大縮小拖拽功能互不影響 1.以vue3中el-carousel輪播插件為例 <div class"pic_view"><el-carousel height"100vh" :autoplay"false" ref"carouselRef" change"handleCarouselChange&qu…

traceroute 使用說明

1、概述 Traceroute&#xff08;Windows 系統中為 tracert&#xff09;是一種網絡診斷工具&#xff0c;用于跟蹤數據包從本地設備到目標主機的傳輸路徑&#xff0c;并顯示沿途經過的每一跳&#xff08;路由器&#xff09;的延遲和 IP 地址。它通過發送不同 TTL&#xff08;生存…

用idea操作git緩存區回退、本地庫回退、遠程庫回退

前言 使用idea軟件操作git非常人性化和方便。 但是如果我的代碼使用git提交之后,我想回到以前的版本,此時需要進行git的版本回退。 提交代碼分為提交到緩存區、本地庫、遠程庫這3個過程。 下面我將介紹每個階段的提交對應的回退方法。 本篇文章是掌握git和使用idea操作git…

webpack+vite前端構建工具 - 3webpack處理js

3 webpack處理js webpack的核心——處理js文件&#xff0c;將模塊化的代碼打包。具體操作如下 es6轉化&#xff08;為兼容老瀏覽器&#xff0c;將es6轉化為es5&#xff09; babel-loader 代碼規范&#xff08;例如空格&#xff0c;縮進等代碼風格規范&#xff09; eslint 代碼…

Nginx轉發中相對路徑資源302問題的分析與解決

Nginx轉發中相對路徑資源302問題的分析與解決 典型案例&#xff1a;后端頁面引入./test.css的302問題 問題場景 假設我們有一個后端服務&#xff0c;其頁面中通過相對路徑引入了CSS文件&#xff1a; <!-- 后端頁面代碼 --> <link rel"stylesheet" href&…