javascript快速初始化數組

来源:https://www.cnblogs.com/ctxb/archive/2022/06/29/16424899.html
-Advertisement-
Play Games

編寫代碼中通常會有快速初始化數組的需求,例如我們需要一個類似matlab里的zeros函數,假如這裡我們需要生成一個0-23的數組用於表示一天24小時。 最基本的做法如下: function(){ let hours = []; for(let k = 0; k < 24; k++ )hours.p ...


編寫代碼中通常會有快速初始化數組的需求,例如我們需要一個類似matlab里的zeros函數,假如這裡我們需要生成一個0-23的數組用於表示一天24小時。
最基本的做法如下:

function(){
    let hours = [];
    for(let k = 0; k < 24; k++ )hours.push(k);
    return hours;
}

下麵我們來思考如何用更優雅的方式實現。
考慮使用new Array(24)+ map的方法來實現。
代碼如下:

Array(24).map((_, h) => h);

註意,這裡map的第二個參數是索引,平時用的少,這裡把索引作為數值。
結果與預期並不符合,為啥呢?
簡單搜索了一下,發現時因為js里的稀疏數組的邏輯導致的。
我們先看一下下麵的代碼:

let a = [];
a[1000] = 2;
console.log(a.length);
// 1000
a.forEach(x => console.log("hello"));
// only one "hello"

js的處理邏輯是,對於沒有主動賦值的位置進行“空置”處理,對於這些“空置”未知,迭代器是不會理會的,這麼做最主要的目的就是避免不合理的賦值操作導致的bug。
假設沒有這種邏輯,我們寫下了new Array(Date.now()),這將導致系統新建一個非常大的數組,而實際上啥也沒存。
我們可以吧new Array(len)乾的事情簡單理解為下麵的過程:

function(len){
    let r = [];
    r.length = len;
    return r;

這就是為什麼對new Array(len)調用map或者forEach的時候跟預期不一致了。
如何解決這一問題呢,除了使用new Array(len)的形式,我們還可以使用new Array(1,2,3)這種寫法來初始化數組,但是這麼寫就沒法實現我們編寫初始化數組的目的了。
這個時候我們想到了apply,這個函數的第二個參數正好就是一個數組,於是我們寫下了下麵的代碼。

// 借用apply
Array.apply(null, Array(24)).map((_, h) => h);
// [0, 1, ..., 24]

得到了我們希望的結果。這就說明,Array(24)apply中作為參數的時候是被當做24個值對待的,因為這一點就保證了最後得到的數組長度是24。
既然如此,我們當然同樣可以使用apply的姊妹函數call

// 借用call
Array.call(null, ...Array(24)).map((_, h) => h);
// [0, 1... 23]

這也更確認了一件事,Array(len)解構會得到len個參數而非一個參數,當然call的使用必須在支持解構操作符的環境中。

在熟悉了callapply的原理後,我們可以進一步寫出下麵的代碼:

// Array本身
Array(...Array(24)).map((_, h) => h);
// [0, 1, ..., 24]

這種形式已經足夠優雅了。
另外,在ES6中,Array提供了新方法fill,借用該方法填充那些“空置”位,進而保證後續的操作能順利進行。
具體代碼如下:

// 推薦
Array(24).fill(null).map((_, h) => h);

現在也比較推薦最後一種寫法,這種方法也最為直觀。
不過需要註意fill方法的使用,應該儘量避免盲目地填充,因為這樣會上面提到了js設計“空置”邏輯為了避免的bug。
有興趣的可以嘗試一下下麵的代碼:

// no-fill
console.time("no-fill");
let t = Array(5e7);
console.timeEnd("no-fill");
// fill
console.time("fill");
let q = Array(5e7).fill(null);
console.timeEnd("fill");
// => no-fill: 0.240ms
// => fill: 3247.921ms

在瀏覽器中嘗試瀏覽器基本會不響應,沒有設置更大的數值為了避免得不到結果。假設一個int占用4個位元組的記憶體,這個fill會占用200M的記憶體,且需要迴圈5千萬次,這必然會占用大量的CPU和記憶體,在單線程的js中是絕對不允許的。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2022年6月28日,HDD·HMS Core.Sparkle影音娛樂沙龍線上上與開發者們見面。HMS Core音頻編輯服務(Audio Editor Kit)專家為大家詳細分享了基於分離的3D音樂創作技術,引起了行業及音樂愛好者的關註。 音頻3D化是音頻發展的重要方向,但當前從源頭製作3D音頻成本 ...
  • 華為帳號是打開華為生態服務的一把鑰匙,只需一個華為帳號,即可在華為全場景設備上使用華為應用市場、華為音樂、華為雲空間、華為錢包等眾多優質的華為應用服務。同時,華為帳號綁定的所有會員權益、購買記錄、收藏/喜歡等數據,也會在不同設備間一鍵同步,輕鬆實現自由切換。 登錄華為帳號後,您可以在多台設備上,快捷 ...
  • Javascript趣味程式從需求到實現-體重測量器 Version 0.0.1 一、需求梳理 二、分析結果: 1、需要提供輸入身高、體重、以及選擇性別的HTML組件,如input、select 2、需要對信息進行驗證,專門的工具類完成這個職責如 “BmiUtils” 3、無效的信息需要提示 4、與 ...
  • 一、效果圖 二、實現邏輯 將需要動態添加的表單項項的綁定值存為一個數組 以迴圈的方式展示form表單 點擊`+`按鈕觸發事件,向數組中新加一個item 點擊`-`按鈕觸發事件,根據迴圈的得到的index來刪除數組中相對應位置的item 三、代碼實現 <template> <div> <el-butt ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 前言 JS是一門單線程語言,單線程就意味著,所有的任務需要排隊,前一個任務結束,才會執行下一個任務。這樣所導致的問題是:如果JS執行的時間過長,這樣就會造成頁面的渲染不連貫,導致頁面渲染載入阻塞的覺。為瞭解決這個問題,JS中出現了同步和異 ...
  • 一、效果圖 二、代碼實現 註:需要安裝依賴 pnpm i sortablejs -S <template> <div style="padding: 15px"> <h3 style="text-align: left; padding: 10px; background: #ccc"> vue+e ...
  • 1. Text 的絕對居中 Android中顯示時會有預設的padding保留,導致垂直居中會有誤差 //不寫下麵兩個,Android系統上文字會偏下* includeFontPadding: false, textAlignVertical: "center", Text省略號顯示 <Text n ...
  • 2.兩數相加 給你兩個 非空 的鏈表,表示兩個非負的整數。它們每位數字都是按照 逆序 的方式存儲的,並且每個節點只能存儲 一位 數字。 請你將兩個數相加,並以相同形式返回一個表示和的鏈表。 你可以假設除了數字 0 之外,這兩個數都不會以 0 開頭。 示例 1: 輸入:l1 = [2,4,3], l2 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...