JavaScript引用类型--Array类型

前言

ECMAScript数组的每一项可以保存任何类型的数据。而且,数组的大小事可以动态调整的,即可以随着数据的添加自动增长以容纳新增数据。

基础知识

  • 创建数组的基本方式有两种。第一种是使用Array构造函数,如下所示:
var colors=new Array();

如果预先知道要保存的项目数量,也可以传递该数量,而该数量会自动变成length属性的值:

var colors=new Array(20);

也可向Array构造函数传递数组中应该包含的项:

var colors=new Array("red","blue","green");
  • 创建数组的第二种基本方式是使用数组字面量表示法:
var colors=["red","blue","green"];//创建一个包含3个字符串的数组
var names=[];//创建一个空数组
var values=[1,2,];
var options=[,,,,,];

以上代码的第一行创建了一个包含3个字符串的数组。第二行使用一对空方括号创建一个空数组。第三行展示了在数组字面量的最后一项听阿金逗号的结果:在IE中,values会成为一个包含3个项且每项的值分别为1、2和undefined的数组;在其他浏览器中,values会成为一个包含2项且值分别为1和2的数组。原因是IE8及以前的ECMAScript实现在数组字面量方面存bug。由于这个bug导致的另一种情况如最后一行代码所示,该行代码可能会创建包含5项的数组(在IE9+、Firefox、Opera、Safari、Chrome中),也可能会创建包含6项的数组(在IE8及更早的版本中)。在像这种省略值的情况下,每一项都将获得undefined值;这个结果与调用Array构造函数时传递项数在逻辑上是相同的,但是由于IE的实现与其他浏览器不一致,因此建议不要使用该语法。
在读取和设置数组的值时,要使用方括号并提供相应的基于0的数字索引,如下:

var colors=["red","blue","green"];
alert(colors[0]);//显示第一项
colors[2]="black";//修改第三项
colors[3]="brown";//新增第四项
alert(colors.length);//4

方括号中的索引表示要访问的值。如果索引小于数组的项数,则返回对应项的值。如果设置某个值超过了现有的项数,数组就会自动增加到该索引值加1的长度。数组的项数保存在length属性中,可通过该属性,可以从数组的末尾移除或向数组中添加新项。

var colors=["red","blue","green"];
colors.length=2;
alert(colors[2]);//undefined
colors.length=4;
alert(colors[3]);//undefined

该数组中一开始有3个值。将length属性设置为2之后会移除一项,结果再访问colors[2]会显示undefined。如果将length属性设置为大于数组项数的值,则新增的每一项都会取得undefined值。
利用length属性也可以在数组末尾添加新项,如下所示:

var colors=["red","blue","green"];
colors[colors.length]="black";//在位置3添加一种颜色
colors[colors.length]="brown";//在位置4添加一种颜色

检测数组

可用两种方法检测某个对象是否为数组:instanceof和Array.isArray()方法。前者当存在两个以上不同的全局执行环境(包含多个框架),那么传入的数组在第二个框架中创建的数组分别具不同的构造函数;后者不支持IE8及以下浏览器。

转换方法

所有对象都具有toLacalString()、toString()和valueOf()方法。
toString:返回由数组中每个值的字符串形式拼接而成的一个以逗号分隔的字符串。
valueOf:返回最适合该对象类型的原始值;也就是是说返回的还是数组。
toLacalString:返回相同的值,但底部调用的是toLacalString方法而不是toString方法。
使用join()方法,则可以使用不同的分隔符来构建这个字符串。

var colors=["red","grren","blue"];
alert(colors.join(","));//red,green,blue
alert(colors.join("||"));//red||green||blue

join()在传递逗号或默认的情况下,和toString()方法一样。

栈方法

栈是一种LIFO(Last-In-First-Out,后进先出)的数据结构,也就是最新添加的项最早被移除。只发生在一个位置——栈的顶部。ECMAScript为数组提供了push()和pop()方法。
push():可接收任意数量的参数,把它们逐个添加到数组末尾,并返回修改后的数组的长度。
pop():从数组末尾移除最后一项,减少length值,然后返回移除的项。

var colors=new Array();//创建一个数组
var count=colors.push("red","green");//推入两项
alert(count);//2

count=colors.push("black");//推入另一项
alert(count);//3

var item=colors.pop();//取得最后一项
alert(item);//"black"
alert(colors.length);//2

队列方法

队列是一张FIFO(First-In-First-Out,先进先出)。队列在列表的末端添加项,从列表的前端移除项。实现这一操作的方法就是shift(),它能够移除数组中的第一个项,并返回该项。

var colors=new Array();//创建一个数组
var count=colors.push("red","green");//推入两项
alert(count);//2

count=colors.push("black");//推入另一项
alert(count);//3

var item=colors.shift();//取得第一项
alert(item);//"red"
alert(colors.length);//2

ECMAScript还为数组提供了unshift()方法,顾名思义,两者的用途相反:它能在数组前端添加任意个项并返回新数组的长度。

重排序方法

reverse():反转数组项的顺序
sort():默认情况下,按升序排列数组项。sort方法会调用toString方法,然后比较得到的字符串,以确定如何排序。

var values=[0,1,5,10,15];
values.sort();
alert(values);//0,1,10,15,5

可见,即使例子中值的顺序没有问题,但sort方法也会根据测试字符串的结果改变原来的顺序。因为数值5虽然小于10,“10”则位于“5”的前面。因此sort方法可以接受一个比较的自定义函数作为参数,以便我们知道哪个值位于哪个值前面。

function compare(value1,value2){
    if(value1<value2){
        return -1;
    }else if(value1>value2){
        return 1;
    }else{
        return 0;
    }
}

var values=[0,1,5,10,15];
values.sort(compare);
alert(values);//0,1,5,10,15

操作方法

  • concat():先创建当前数组的一个副本,然后将接收到的参数添加到这个副本的末尾,最后返回新构建的数组。在每有给concat方法传递参数的情况下,它只是复制当前数组并返回副本。如果传递给concat方法的是一个或多个数组,则该方法会将这些数组中的每一项都添加到结果数组中。如果传递的值不是数组,这些值就会被简单地添加到结果数据的末尾。示例如下:
var colors=["red","green","blue"];
var colors2=colors.concat("yellow",["black","brown"]);

alert(clors);//red,green,blue
alert(colors2);//red,green,blue,yellow,black,brown
  • slice():基于当前数组中的一或多个项创建一个新数组。可接受一或两个参数,即要返回项的起始和结束位置。在一个参数下,返回从该参数指定位置开始到当前数组末尾的所有项。如果是两个参数,返回起始和结束位置之间的项,但不包括结束位置的项。该方法不会影响原始数组。示例如下:
var colors=["red","green","blue""yellow","purple"];
var colors2=colors.slice(1);
var colors3=colors.slice(1,4);

alert(colors2);//green,blue,yellow,purple
alert(colors3);//green,blue,yellow
  • splice():主要用途是向数组的中部插入项,方式如下:
    1.删除:可以删除任意数量的项,只需两个参数:要删除的第一项的位置和要删除的项数。例如:splice(0,2)会删除数组的前两项。
    2.插入:可以向指定位置插入任意数量的项,只需提供3个参数:起始位置、0(要删除的项数)和要插入的项。如果要插入多个项,可以再传入第四、第五、以至任意多个项,例如:splice(2,0,”red”,”green”)会从当前数组的位置2开始插入字符串。
    3.替换:可以向指定位置插入任意数量的项,且同时删除任意数量的项,就是把插入的0(要删除的项)改为要删除的任意数量的项
var colors=["red","green","blue"];
var removed=colors.splice(0,1);//删除第一项
alert(colors);//green,blue
alert(removed);//red

removed=colors.splice(1,0,"yellow","orange");//从位置1开始插入两项
alert(colors);//green,yellow,orange,blue
alert(removed);返回的是一个空数组

removed=colors.splice(1,1,"red","purple");//插入两项,删除一项
alert(colors);//green,red,purple,orange,blue
alert(removed);//yellow

位置方法

indexOf()和lastIndexOf()。这两个方法都接收两个参数:要查找的项和(可选)表示查找起点位置的索引。其中,前者从数组的开头(位置0)开始向后查找,后者则从数组的末尾向前查找。这两个方法都返回要查找的项在数组的位置。

var numbers=[1,2,3,4,5,4,3,2,1];
alert(numbers.indexOf(4));//3
alert(numbers.lastIndexOf(4));//5

alert(numbers.indexOf(4,4));//5
alert(numbers.lastIndexOf(4,4));//3

迭代方法

ECMAScript为数组定义了5个迭代方法。传入这些方法中的函数会接收三个参数:数组项的值、该项在数组中的位置和数组对象本身。

  • every():对数组中的每一项运行给定函数,如果该函数对每一项都返回true,则返回true。
  • filter():对数组中的每一项运行给定函数,返回该函数会返回true的项组成的数组。
  • forEach():对数组中的每一项运行给定函数。这个函数没有返回值。
  • map():对数组中的每一项运行给定函数,返回每次函数调用的结果组成的数组。
  • some():对数组中的每一项运行给定函数,如果该函数对任一项返回true。则返回true。
    这些方法中,最相似的是every()和some(),它们都用于查询数组中的项是否满足某个条件。对every()来说,传入的函数必须对每一项都返回true,这个方法才返回true。而some()方法则只要传入的函数对数组中的某一项返回true,就会返回true。示例如下:
var numbers=[1,2,3,4,5,4,3,2,1];
var everyResult=numbers.every(function(item,index,arry){
    return (item>2);
});
alert(everyRusult);//false
var someResult=numbers.some(function(item,index,arry){
    return (item>2);
});
alert(someResult);//true

下面看一下filter()函数:

var numbers=[1,2,3,4,5,4,3,2,1];
var filterResult=numbers.filter(function(item,index,arry){
    return (item>2);
});
alert(filterResult);//[3,4,5,4,3]

map()也返回一个数组,而这个数组的每一项都是在原始数组中的对应项上运行传入函数的结果。例如,可以给数组中的每一项乘以2,然后返回这些乘积组成的数组,如下所示:

var numbers=[1,2,3,4,5,4,3,2,1];
var mapResult=numbers.map(function(item,index,arry){
    return item*2;
});
alert(mapResult);//[2,4,6,8,10,8,6,4,2]

最后一个方法四forEach(),它只是对数组中的每一项运行传入的函数。这个方法没有返回值,本质上与使用for循环迭代数组一样,如下所示:

var numbers=[1,2,3,4,5,4,3,2,1];
var forEachResult=numbers.forEach(function(item,index,arry){
    //执行某些操作
});

IE8及以下不支持这些迭代方法。

归并方法

reduce()和reduceRight()。这两个方法都会迭代数组中所有项,然后构建一个最终返回的值。其中。reduce()方法从数组的第一项开始,逐个遍历到最后。而reduceRight()则从数组的最后一项开始,向前遍历到第一项。
这两个方法都接收两个参数:一个在每一项上调用的函数和(可选)作为归并基础的初始值。传入给两个方法的函数接受4个参数:前一个值、当前值、项的索引和数组对象。这个函数返回的任何值都会作为第一个参数自动传个下一项。第一次迭代发生在数组的第二项上,因此第一个参数是数组的第一项,第二个参数是数组的第二项。
使用reduce()方法可以指向求数组中所有值之和的操作,比如:

var values=[1,2,3,4,5];
var sum=values.reduce(function(prev,cur,index,array){
    return prev+cur;
});
alert(sum);//15

第一次指向回调函数,prev是1,cur是2。第二次,prev是3(1加2的结果),cur是3(数组的第三项)。这个过程会持续到把数组中的每一项都访问一遍,最后返回结果。
reduceRight()的作用类似,只不过方向相反

var values=[1,2,3,4,5];
var sum=values.reduceRight(function(prev,cur,index,array){
    return prev+cur;
});
alert(sum);//15

在这个例子中,第一次指向回调函数,prev是5,cur是4。使用reduce()还是reduceRight(),主要取决于要从哪头开始遍历数组。除此之外,它们完全相同。
IE8及以下不支持这两个归并方法。

总结

下面的思维导图是总结了Javascript中的Array类型在本文中提到的属性及方法。
这里写图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值