JavaScript 特殊對象 Array-Like Objects 詳解

這篇文章拖了有兩周凌埂,今天來跟大家聊聊 JavaScript 中一類特殊的對象 -> Array-Like Objects。

(本文節(jié)選自 underscore 源碼解讀系列文章拆祈,完整版請關注 https://github.com/hanzichi/underscore-analysis

Array-Like

JavaScript 中一切皆為對象,那么什么是 Array-Like Objects倘感?顧名思義放坏,就是像數(shù)組的對象,當然老玛,數(shù)組本身就是對象嘛淤年!稍微有點基礎的同學钧敞,一定知道 arguments 就是 Array-Like Objects 的一種,能像數(shù)組一樣用 [] 去訪問 arguments 的元素麸粮,有 length 屬性溉苛,但是卻不能用一些數(shù)組的方法,如 push弄诲,pop愚战,等等。

那么齐遵,什么樣的元素是 Array-Like Objects寂玲?我們來看看 underscore 中對其的定義。

var MAX_ARRAY_INDEX = Math.pow(2, 53) - 1;
var getLength = property('length');
var isArrayLike = function(collection) {
  var length = getLength(collection);
  return typeof length == 'number' && length >= 0 && length <= MAX_ARRAY_INDEX;
};

很簡單梗摇,不是數(shù)組拓哟,但是有 length 屬性尔许,且屬性值為非負 Number 類型即可广料。至于 length 屬性的值汁雷,underscore 給出了一個上限值 MAX_ARRAY_INDEX茅特,其實是 MAX_SAFE_INTEGER(感謝 @HangYang 同學指出) ,因為這是 JavaScript 中能精確表示的最大數(shù)字春瞬。

想想還有什么同時能滿足以上條件的?NodeList,HTML Collections景图,仔細想想,甚至還有字符串碉哑,或者擁有 length 屬性的對象挚币,函數(shù)(length 屬性值為形參數(shù)量),等等扣典。

Array-Like to Array

有的時候妆毕,需要將 Array-Like Objects 轉為 Array 類型,使之能用數(shù)組的一些方法贮尖,一個非常簡單粗暴并且兼容性良好的方法是新建個數(shù)組笛粘,然后循環(huán)存入數(shù)據(jù)。

我們以 arguments 為例湿硝。

function fn() {
  // Uncaught TypeError: arguments.push is not a function
  // arguments.push(4);

  var arr = [];
  for (var i = 0, len = arguments.length; i < len; i++)
    arr[i] = arguments[i];

  arr.push(4); // [1, 2, 3, 4]
}

fn(1, 2, 3);

但是這不是最優(yōu)雅的薪前,更優(yōu)雅的解法大家一定都知道了,use Array.prototype.slice(IE9- 會有問題)关斜。

function fn() {
  var arr = Array.prototype.slice.call(arguments);
  arr.push(4); // arr -> [1, 2, 3, 4]
}

fn(1, 2, 3);

或者可以用 [] 代替 Array.prototype 節(jié)省幾個字節(jié)示括。

function fn() {
  var arr = [].slice.call(arguments);
  arr.push(4); // arr -> [1, 2, 3, 4]
}

fn(1, 2, 3);

如果非得追求性能,用 [] 會新建個數(shù)組痢畜,性能肯定不及前者垛膝,但是由于引擎的優(yōu)化鳍侣,這點差異基本可以忽略不計了(所以很多框架用的就是后者)。

為什么這樣可以轉換吼拥?我們簡單了解下倚聚,主要的原因是 slice 方法只需要參數(shù)有 length 屬性即可。首先扔罪,slice 方法得到的結果是一個 新的數(shù)組秉沼,通過 Array.prototype.slice.call 傳入的參數(shù)(假設為 a),如果沒有 length 屬性矿酵,或者 length 屬性值不是 Number 類型唬复,或者為負,那么直接返回一個空數(shù)組全肮,否則返回 a[0]-a[length-1] 組成的數(shù)組敞咧。(具體可以看下 v8 源碼 https://github.com/v8/v8/blob/master/src/js/array.js#L621-L660

當然,ES6 提供了更簡便的方法辜腺。

var str = "helloworld";
var arr = Array.from(str); 
// ["h", "e", "l", "l", "o", "w", "o", "r", "l", "d"]

小結下休建,如果要把 Array-Like Objects 轉為 Array,首選 Array.prototype.slice评疗,但是由于 IE 下 Array.prototype.slice.call(nodes) 會拋出錯誤(because a DOM NodeList is not a JavaScript object)测砂,所以兼容的寫法如下。(但還有一點要注意的是百匆,如果是 arguments 轉為 Array砌些,最好別用 Array.prototype.slice,V8 下會很慢加匈,具體可以看下 避免修改和傳遞 arguments 給其他方法 — 影響優(yōu)化

function nodeListToArray(nodes){
  var arr, length;

  try {
    // works in every browser except IE
    arr = [].slice.call(nodes);
    return arr;
  } catch(err){
    // slower, but works in IE
    arr = [];
    length = nodes.length;

    for(var i = 0; i < length; i++){
       arr.push(nodes[i]);
     }  

    return arr;
  }
} 

Others

很多時候存璃,某個方法你以為接收的參數(shù)是數(shù)組,其實類數(shù)組也是可以的雕拼。

Function.prototype.apply() 函數(shù)接收的第二個參數(shù)纵东,其實也可以是類數(shù)組。

var obj = {0: 4, length: 2};
var arr = [1, 2, 3];
Array.prototype.push.apply(arr, obj);
console.log(arr); // [1, 2, 3, 4, undefined]

Read More

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末啥寇,一起剝皮案震驚了整個濱河市偎球,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌辑甜,老刑警劉巖甜橱,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異栈戳,居然都是意外死亡岂傲,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進店門子檀,熙熙樓的掌柜王于貴愁眉苦臉地迎上來镊掖,“玉大人乃戈,你說我怎么就攤上這事∧督” “怎么了症虑?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長归薛。 經(jīng)常有香客問我谍憔,道長,這世上最難降的妖魔是什么主籍? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任习贫,我火速辦了婚禮,結果婚禮上千元,老公的妹妹穿的比我還像新娘苫昌。我一直安慰自己,他們只是感情好幸海,可當我...
    茶點故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布祟身。 她就那樣靜靜地躺著,像睡著了一般物独。 火紅的嫁衣襯著肌膚如雪袜硫。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天挡篓,我揣著相機與錄音父款,去河邊找鬼。 笑死瞻凤,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的世杀。 我是一名探鬼主播阀参,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼瞻坝!你這毒婦竟也來了蛛壳?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤所刀,失蹤者是張志新(化名)和其女友劉穎衙荐,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體浮创,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡忧吟,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了斩披。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片溜族。...
    茶點故事閱讀 39,690評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡讹俊,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出煌抒,到底是詐尸還是另有隱情仍劈,我是刑警寧澤,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布寡壮,位于F島的核電站贩疙,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏况既。R本人自食惡果不足惜这溅,卻給世界環(huán)境...
    茶點故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望坏挠。 院中可真熱鬧芍躏,春花似錦、人聲如沸降狠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽榜配。三九已至否纬,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間蛋褥,已是汗流浹背临燃。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留烙心,地道東北人膜廊。 一個月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像淫茵,于是被迫代替她去往敵國和親爪瓜。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,577評論 2 353

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