基础储备
数组的本质就是一种特殊的对象,使用typeof可以返回数组的类型是Object,然后数组的特殊性在于他的键名是按次序排列的一组整数(0,1,2…)。
JavaScript语言规定,对象的键名一律为字符串,所以,数组的键名其实也是字符串。之所以可以用数值读取,是因为非字符串的键名会被转为字符串。
由于数组成员的键名是固定的,因此数组不用为每个元素指定键名,而对象的每个成员都必须指定键名。
所以我们在上面截图中得到的键名是“0”,“1”;
如果一个值可以被转换为 整数 ,则以该值为键名,等于以对应的整数为键名。
问题
但是当我们用arr['p'] =3;的时候,我们会发现并不会改变arr.length属性
但是我们打印arr的时候就会出现下列结果
这是为什么呢?
查阅了很多资料,然后再找到了关键
- 数组的属性length的基本功能是追踪数组的最大索引
2.length属性表示一个无符号 32-bit 整数,返回一个数组中的元素个数。 - 一个属性键n(假设n为字符串)是某个数组的索引当且仅当ToString(ToUint32(n))等于n且ToUnit32(n)不等于232 -1,
- 数组属性的键是数组索引,这个属性成为元素,也可以理解为数组成员,
- 换句话说,在规范中所有方括号中的值,甚至是数字,都可以转换为字符串且解释为属性的键。
- 属性n(字符串)如果要作为数组索引必须等于把P转换为的数字,等于把数字转换为32位无符号的整数,等于把整型转化为的字符串,即数组的索引必须是字符串化的整型,而且是在0<=i<232-1范围内。
在例子中arr['p']中的p,将‘p'进行转换实验
因为,** 属性n(字符串)如果要作为数组索引必须等于(把P转换为数字,把数字转换为32位无符号整数,把整型转化为字符串)**,但是上面实验截图现实'p'即不等于将'p'转换的数字,也不等于将这个数字转换了的32位无符号整数,也不等于将这个整型转换为的字符串。所以p不能作为数组索引,而只是数组的一个属性了。因此当属性n(字符串)不是一个字符串化的整型时,或者不在范围0<=i<232-1范围内时,他的效果就是数组的一个属性(数组也是一个对象,这个n只是该对象的一个普通属性),
因此,解释到这里,就能理解为什么当使用arr['p'] =3的时候,并不会改变arr.length属性。因为arr['p']不是数组元素,而length是返回数组的元素个数的属性,索引增加arr['p']不会改变length的长度。
小知识
Array对象的length属性是一个数据属性,其值始终大于每个可删除属性的名称,其名称为数组索引。
length属性最初具有属性{[[Writable]]:true,[[Enumerable]]:false,[[Configurable]]:false}。