博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
深入理解JavaScript类数组
阅读量:6404 次
发布时间:2019-06-23

本文共 3579 字,大约阅读时间需要 11 分钟。

起因

写这篇博客的起因,是我在知乎上回答一个问题时,说自己在学前端时把《JavaScript高级程序设计》看了好几遍。

于是在评论区中,出现了如下的对话:
对话

天啦噜,这话说的,宝宝感觉到的,是满满的恶意啊。还好自己的JavaScript基础还算不错,没被打脸。(吐槽一句:知乎少部分人真的是恶意度爆表,整天想着打别人的脸。都是搞技术的,和善一点不行吗…………)

不过这个话题也引起了我的注意,问了问身边很多前端同学关于数组与类数组的区别。他们都表示不太熟悉,所以决定写一篇博客,来分享我对数组与类数组的理解。

什么是类数组

类数组的定义,有如下两条:

  • 具有:指向对象元素的数字索引下标以及 length 属性告诉我们对象的元素个数

  • 不具有:诸如 push 、 forEach 以及 indexOf 等数组对象具有的方法Q

这儿有三个典型的JavaScript类数组例子。

1. DOM方法

// 获取所有divlet arrayLike = document.querySelectorAll('div')console.log(Object.prototype.toString.call(arrayLike))  // [object NodeList]console.log(arrayLike.length) // 127console.log(arrayLike[0]) // 
console.log(Array.isArray(arrayLike)) // falsearrayLike.push('push') // Uncaught TypeError: arrayLike.push is not a function(…)

是的,这个arrayLike的 NodeList,有length,也能用数组下标访问,但是使用Array.isArray测试时,却告诉我们它不是数组。直接使用push方法时,当然也会报错。

但是,我们可以借用类数组方法:

let arr = Array.prototype.slice.call(arrayLike, 0)console.log(Array.isArray(arr)) // truearr.push('push something to arr')console.log(arr[arr.length - 1]) // push something to arr

不难看出,此时的arrayLike在调用数组原型方法时,返回值已经转化成数组了。也能正常使用数组的方法。

2. 类数组对象

let arrayLikeObj = {  length: 2,  0: 'This is Array Like Object',  1: true}console.log(arrayLikeObj.length) // 2console.log(arrayLikeObj[0]) // This is Array Like Objectconsole.log(Array.isArray(arrayLikeObj)) // falselet arrObj = Array.prototype.slice.call(arrayLikeObj, 0)console.log(Array.isArray(arrObj)) // true

这个例子也很好理解。一个对象,加入了length属性,再用Array的原型方法处理一下,摇身一变成为了真的数组。

3. 类数组函数

这个应该算是最好玩,也是最迷惑人的类数组对象了。

let arrayLikeFunc1 = function () {}console.log(arrayLikeFunc1.length) // 0let arrFunc1 = Array.prototype.slice.call(arrayLikeFunc1, 0)console.log(arrFunc1, arrFunc1.length) // ([], 0)let arrayLikeFunc2 = function (a, b) {}console.log(arrayLikeFunc2.length) // 2let arrFunc2 = Array.prototype.slice.call(arrayLikeFunc2, 0)console.log(arrFunc2, arrFunc2.length) // ([undefined × 2], 2)

可以看出,函数也有length属性,其值等于函数要接收的参数。

注:不适用于ES6的rest参数。具体原因和表现这儿就不再阐述了,不属于本文讨论范围。可参见 。另外arguments在ES6中,被rest参数代替了,所以这儿不作为例子。

而length属性大于0时,如果转为数组,则数组里的值会是undefined。个数等于函数length的长度。

类数组的实现原理

类数组的实现原理,主要有以下两点:

第一点是JavaScript的“万物皆对象”概念。
第二点则是JavaScript支持的“鸭子类型”。

首先,从第一点开始解释。

万物皆对象

万物皆对象具体解释如下:

在JavaScript中,“一切皆对象”,数组和函数本质上都是对象,就连三种原始类型的值——数值、字符串、布尔值——在一定条件下,也会自动转为对象,也就是原始类型的“包装对象”。

而另外一个要点则是,所有对象都继承于Object。所以都能调用对象的方法,比如使用点和方括号访问属性。

比如说,这样的:

let func = function() {}console.log(func instanceof Object) // truefunc[0] = 'I\'m a func'console.log(func[0]) // 'I\'m a func'

鸭子类型

万物皆对象具体解释如下:

如果它走起来像鸭子,而且叫起来像鸭子,那么它就是鸭子。

比如说上面举的类数组例子,虽然他们是对象/函数,但是只要有length属性和对应的数字下标,那么他们就是数组。

但是,在这儿,还是有些迷糊的。为什么使用call/apply借用数组方法就能处理这些类数组呢?

探秘V8

一开始,我也对这个犯迷糊啊。直到我去Github上,看到了谷歌V8引擎处理数组的源代码。

地址在这儿:
作为讲述,我们在这里引用push的源代码(方便讲述,删除部分。slice的比较长,但是原理一致):

// Appends the arguments to the end of the array and returns the new// length of the array. See ECMA-262, section 15.4.4.7.function ArrayPush() {  // 获取要处理的数组  var array = TO_OBJECT(this);  // 获取数组长度  var n = TO_LENGTH(array.length);  // 获取函数参数长度  var m = arguments.length;  for (var i = 0; i < m; i++) {    // 将函数参数push进数组    array[i+n] = arguments[i];  }  // 修正数组长度  var new_length = n + m;  array.length = new_length;  // 返回值是数组的长度  return new_length;}

是的,整个push函数,并没有涉及是否是数组的问题。只关心了length。而因为其对象的特性,所以可以使用方括号来设置属性。

这也是万物皆类型和鸭子类型最生动的体现。

总结

JavaScript中的类数组的特殊性,是由其“万物皆类型”和“鸭子类型”决定的,而浏览器引擎底层的实现,更是佐证了这一点。

而先前说我的那位同学,因为只是知道类数组的几种表现和用法,并且想通过apply来打我脸,证明我根本没有仔细看书。这种行为不仅不友善,而且学习效率也不高。
因为,知其然而不知其所以然是不可取的。特别是发现很多这种例子,就得学会归纳总结。(感谢winter老师的演讲:,教会我很多东西。)。
很多时候,深入看看源代码也会让你对这个理解的更透彻。将来就算是蹦出一百种类数组,也能知道是怎么回事儿。

最后,还是开头那句话:“都是搞技术的,和善一点不行吗?有问题就好好交流,不要总想着打别人脸啊…………”

最后附上本人博客地址和原文链接,希望能与各位多多交流。

原文链接:

转载地址:http://phnea.baihongyu.com/

你可能感兴趣的文章
python 生成html代码_使用Python Markdown 生成 html
查看>>
axure如何导出原件_Axure 教程:轻松导出图标字体所有图标
查看>>
laravel input值必须不等于0_框架不提供,动手造一个:Laravel表单验证自定义用法...
查看>>
cad填充图案乱理石_太快了吧!原来大神是这样用CAD图案填充的
查看>>
activator.createinstance 需要垃圾回收么_在垃圾回收器中有哪几种判断是否需要被回收的方法...
查看>>
rocketmq 消息指定_RocketMQ入坑系列(一)角色介绍及基本使用
查看>>
redis zset转set 反序列化失败_掌握好Redis的数据类型,面试心里有底了
查看>>
p图软件pⅰc_娱乐圈最塑料的夫妻,P图永远只P自己,太精彩了吧!
查看>>
jenkins 手动执行_Jenkins 入门
查看>>
怎么判断冠词用a还是an_葡语干货 | 葡萄牙语冠词用法整理大全
查看>>
js传参不是数字_JS的Reflect学习和应用
查看>>
三个不等_数学一轮复习05,从函数观点看方程与不等式,记住口诀与联系
查看>>
右键新建文件夹_Macos电脑鼠标右键木有新建文档咋办,有办法,莫捉急
查看>>
卡尺测量的最小范围_汽车维修工具-测量用具
查看>>
网优5g前景_5G网络优化师前景怎么样?
查看>>
竞态条件的赋值_[译] part25: golang Mutex互斥锁
查看>>
delmatch oracle_完美完全卸载(清除)oracle数据库的方式(方法)
查看>>
pyqt 滚动条 美化_Pyqt5 关于流式布局和滚动条的综合使用示例代码
查看>>
51单机片 编译hex_单片机爬坑记-05-编译环境(完)
查看>>
java 正则表达式 img_Java正则表达式获得html字符串里的<img src=""/> 中的url列表
查看>>