從資料結構分析看:用for each...in 比 for...in 要快些

來源:互聯網
上載者:User

之前聽說Firefox的JS引擎支援for each in的文法,例如下述的代碼:

複製代碼 代碼如下:
var arr = [10,20,30,40,50];
for each(var k in arr)
    console.log(k);

即可直接遍曆出arr數組的內容。

由於只有FireFox才支援,所以幾乎所有的JS代碼都不用這一特徵。

不過在ActionScript裡天生就支援for each的文法,不論Array還是Vector,還是Dictionary,只要是可枚舉的對象都可以for in和for each in。

之前並沒有感覺有太大的差異,為了懶得敲一個each單詞,一直用熟悉的for in來遍曆。

不過今天仔細琢磨了會,從資料結構的角度分析了下,覺得for in和for each in效率上有著本質的區別,無論是JS還是AS。

原因很簡單:Array不是真正意義上的數組!

何為真正意義的數組?當然就是傳統語言裡type[]定義的資料類型,所有元素都是連續儲存的。

“Array”雖然也是數組的意思,但熟悉JS的都知道,它其實是個非線性偽數組,下標可以是任一數字。寫入arr[1000000]並非真正申請容納一百萬個元素的空間,而是把1000000轉換成相應的雜湊值,對應到很小一塊儲存空間裡,從而節省了大量記憶體。

例如有如下數組:

複製代碼 代碼如下:
var arr = [];
  arr[10] = 1000;
  arr[20] = 2000;
  arr[30] = 5000;
  arr[40] = 8000;
  arr[200] = 9000;

用for...in遍曆Array,是個很累贅的過程:

遍曆時每次訪問arr[k],都要進行一次Hash(k)計算,根據散列表的容量模數,最終在衝突鏈表裡找到結果。

如果支援for each...in的文法,其內部的資料結構就決定了會快很多:

Array裡儲存存了keys的列表,也把每個values值作為鏈表關聯起來。每當有值添加或刪除,就更新其連結關係。

當for each...in遍曆時,只需從第一個節點往後迭代即可,無需任何Hash計算。

當然,對於AS3裡Vector這樣的線性數組來說,兩者相差不大;同理,HTML5裡支援二進位的數組ArrayBuffer也是如此。不過從理論上來看,即使arr是個連續的線性數組,for each in還是要快一點:

for...in遍曆時,每次訪問arr[k]都要進行下標越界檢查;而for each in則根據內部鏈表,直接從底層反饋出迭代變數,節省了越界檢查的過程。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.