JavaScript 配列の 22 の一般的なメソッドの詳細なコードの紹介

黄舟
リリース: 2017-03-06 14:56:05
オリジナル
1396 人が閲覧しました

前述の通り

この記事では、配列には合計 22 のメソッドがあり、それらをオブジェクト継承メソッド、配列変換メソッド、スタックおよびキュー メソッド、配列ソート メソッド、配列スプライシング メソッド、サブ配列作成メソッド、配列削除に分けます。位置メソッド、配列結合メソッド、配列反復メソッドの 10 のカテゴリがあり、詳しく説明します

オブジェクト継承メソッド

Array は、toString()、toLocaleString()、および valueOf() を継承する特別なオブジェクトです。 ) オブジェクト Object のメソッド [toString()] toString() メソッドは、配列内の各値の文字列形式で連結されたカンマ区切りの文字列を返します [注] このメソッドの戻り値は、パラメータを指定しない join() メソッドでも返される文字列は同じです

[1,2,3].toString();//'1,2,3'
['a','b','c'].toString();//'a,b,c'
[1,[2,'c']].toString();//'1,2,c'
ログイン後にコピー

alert() は文字列パラメータを受け取る必要があるため、バックグラウンドで toString() メソッドを呼び出し、toString() メソッドと同じ結果を取得します。

alert([1,2,3]);//'1,2,3'
ログイン後にコピー

【toLocaleString()】toLocaleString() は toString() メソッドのローカライズ版で、多くの場合 toString() メソッドと同じ値を返しますが、常に同じ値を返すわけではありません。要素の toLocaleString() メソッドを呼び出して、各配列要素を文字列に変換するためです

var person1 = {
    toLocaleString: function(){
        return 'Nikolaos';
    },
    toString: function(){
        return 'Nicholas';
    }
};
var person2 = {
    toLocaleString: function(){
        return 'Grigorios';
    },
    toString: function(){
        return 'Greg';
    }
};
var people = [person1,person2];
console.log(people.toString());//'Nicholas,Greg'
console.log(people.toLocaleString());//'Nikolaos,Grigorios'
ログイン後にコピー

配列内の項目の値が null または未定義の場合、値は toLocaleString() メソッドと toString() メソッドによって返されます結果は空の文字列で表されます

var colors = [1,undefined,2,null,3];
console.log(colors.toString());//'1,,2,,3'
console.log(colors.toLocaleString());//'1,,2,,3'
ログイン後にコピー

【valueOf()】valueOf()メソッドは配列オブジェクトそのものを返します

var a = [1, 2, 3];
console.log(a.valueOf());// [1, 2, 3]
console.log(a.valueOf() instanceof Array);//true
ログイン後にコピー

配列変換メソッド

【join()】Array.join()メソッドは文字列です. 文字列を複数のブロックに分割して配列を作成する、split() メソッドの逆の操作。配列によって継承される toLocaleString() メソッドと toString() メソッドは、デフォルトで配列項目をカンマ区切り文字の形式で返します。 join() メソッドは、さまざまな区切り文字を使用してこの文字列を構築できます。 join() メソッドは、区切り文字として使用される文字列を 1 つのパラメータだけ受け取り、join() に渡されない場合は、すべての配列項目を含む文字列を返します。メソッド 値を入力する場合は、区切り文字としてカンマを使用してください

var a = [1,2,3];
console.log(a.join());//'1,2,3'
console.log(a.join(' '));//'1 2 3'
console.log(a.join(''));//'123'

var b = new Array(10);
b.join('-');//'---------',9个连字符组成的字符串
ログイン後にコピー

join() メソッドのパラメータが未定義の場合、標準ブラウザは区切り文字としてカンマを使用して文字列を返しますが、IE7 ブラウザは「未定義」を使用します文字列を返すための区切り文字として ' を使用します

//标准浏览器为'1,2,3';IE7-浏览器为'1undefined2undefined3'
var a = [1,2,3];
console.log(a.join(undefined));
ログイン後にコピー

配列内の項目の値が null または未定義の場合、その値は join() メソッドによって返される結果では空の文字列で表されます

var colors = [1,undefined,2,null,3];
console.log(colors.join());//'1,,2,,3'
ログイン後にコピー

このメソッドは次のことができます配列のようなオブジェクトにも使用できます

console.log(Array.prototype.join.call('hello', '-'));// "h-e-l-l-o"
var obj = { 0: 'a', 1: 'b', length: 2 };
console.log(Array.prototype.join.call(obj, '-'));// 'a-b'
ログイン後にコピー

[注意] オブジェクトに長さ属性がない場合、そのオブジェクトは配列クラスではないため、配列メソッドを呼び出すことはできません

var obj = { 0: 'a', 1: 'b' };
console.log(typeof Array.prototype.join.call(obj, '-'));//''
ログイン後にコピー

スタックメソッドやキューメソッド

push()および Pop() メソッドを使用すると、配列をスタックとして使用できます。 unshift() メソッドとshift() メソッドの動作は、push() および Pop() と非常に似ていますが、前者は配列の末尾ではなく先頭に要素を挿入および削除する点が異なります。 LIFO( Last-First-Out (後入れ先出し) データ構造。つまり、最後に追加された項目が最も早く削除されます。スタック内の項目の挿入 (プッシュと呼ばれる) と削除 (ポップと呼ばれる) は、スタックの最上部という 1 つの場所でのみ発生します。 JavaScript は、スタックのような動作を実現するために配列専用の Push() メソッドと Pop() メソッドを提供します。キュー データ構造のアクセス ルールは FIFO (先入れ先出し、先入れ先出し) です。キューはリストの最後に項目を追加し、リストの先頭から項目を削除します。 shift()メソッドとpush()メソッドを組み合わせると、配列をキューのように使用できます

[push()]

push()メソッドは任意の数のパラメータを受け取り、それらを配列の最後に追加します1 ずつ増やして、変更された配列の長さを返します。したがって、この配列は元の配列を変更します

var a = [];
console.log(a,a.push(1));//[1] 1
console.log(a,a.push('a'));//[1,'a'] 2
console.log(a,a.push(true, {}));//[1,'a',true,{}] 4
console.log(a,a.push([5,6]));//[1,'a',true,{},[5,6]] 5
ログイン後にコピー
2つの配列を結合する必要がある場合は、applyメソッドを使用できます

var a = [1, 2, 3];
var b = [4, 5, 6];
console.log(a,Array.prototype.push.apply(a, b));//[1,2,3,4,5,6] 6
ログイン後にコピー

[注意] callメソッドを使用すると、配列b全体がパラメータとみなされます

var a = [1, 2, 3];
var b = [4, 5, 6];
console.log(a,Array.prototype.push.call(a, b));//[1,2,3,[4,5,6]] 4
ログイン後にコピー

push() メソッド オブジェクトに要素を追加することもできます。追加されたオブジェクトは配列のようなオブジェクトになります。つまり、新しく追加された要素のキーは配列のインデックスに対応し、オブジェクトは長さ属性を持ちます。

var obj = {a: 1};
console.log(obj,[].push.call(obj, 2));// {a:1, 0:2, length: 1}
console.log(obj,[].push.call(obj, [3]));// {a:1, 0:2, 1:[3], length: 2}
ログイン後にコピー

【pop()】

pop() このメソッドは、配列の末尾から最後の項目を削除し、配列の長さをデクリメントし、削除された項目を返します。したがって、配列は元の配列を変更します

var a = ['a', 'b', 'c'];
console.log(a,a.pop()); // ['a', 'b'] 'c'
ログイン後にコピー
空の配列でpop()メソッドを使用すると、エラーは報告されませんが、未定義が返されます

var a = [];
console.log(a,a.pop()); // [] undefined
ログイン後にコピー

【shift()】

shift()メソッドは、配列の最初の要素 項目が返され、配列の長さが 1 つ減ります。したがって、配列は元の配列を変更します

var a = ['a', 'b', 'c'];
console.log(a,a.shift());//['b', 'c'] 'a'
ログイン後にコピー
空の配列でshift()メソッドを使用すると、エラーは報告されませんが、未定義が返されます

var a = [];
console.log(a,a.shift());// [] undefined
ログイン後にコピー

【unshift()】

unshift()メソッドは追加します配列項目の前に任意の数の項目を追加し、新しい配列の長さを返します。したがって、配列は元の配列を変更します

var a = ['a', 'b', 'c'];
console.log(a,a.unshift('x')); //['x', 'a', 'b', 'c'] 4
ログイン後にコピー
複数のパラメーターを指定して unshift() が呼び出された場合、パラメーターは一度に 1 つずつではなく、一度に挿入されます。これは、最終的な配列に挿入される要素の順序が、パラメーター リスト内の順序と一致していることを意味します

var a = ['a', 'b', 'c'];
console.log(a,a.unshift('x','y','z')); //['x','y','z','a', 'b', 'c'] 6
ログイン後にコピー

[注] IE7 ブラウザーでは、unshift() メソッドは常に未定義を返します

//标准浏览器下,返回[1] 1;而IE7-浏览器下,返回[1] undefined
var a = [];
console.log(a,a.unshift(1));
ログイン後にコピー

配列の並べ替えメソッド

並べ替えに直接使用できる配列内の 2 つのメソッド: reverse() と sort()

【reverse()】

reverse() メソッドは、配列の順序を反転し、ソートされた配列を返すために使用されます。元の配列の順序も変更されました

var array = [1,2,4,3,5];
console.log(array,array.reverse());//[5,3,4,2,1] [5,3,4,2,1]
var array = ['str',true,3];
console.log(array,array.reverse());//[3,true,'str'] [3,true,'str']
ログイン後にコピー

【sort()】

默认情况下,sort()方法按字符串升序排列数组项,sort方法会调用每个数组项的toString()方法,然后比较得到的字符串排序,返回经过排序之后的数组,而原数组顺序也发生改变

var array = [1,2,4,3,5];
console.log(array,array.sort());//[1,2,3,4,5] [1,2,3,4,5]
var array = ['3str',3,2,'2'];
console.log(array,array.sort());//[2, "2", 3, "3str"] [2, "2", 3, "3str"]
var array = [1,5,10,50];
console.log(array,array.sort());//[1, 10, 5, 50] [1, 10, 5, 50]
ログイン後にコピー

如果数组包含undefined元素,它们会被排到数组的尾部

var array = ['3',3,undefined,2,'2'];
console.log(array,array.sort());//["2", 2, "3", 3, undefined] ["2", 2, "3", 3, undefined]
ログイン後にコピー

sort()方法可以接受一个比较函数作为参数,以便指定哪个值在哪个值的前面。比较函数接收两个参数,如果第一个参数应该位于第二个参数之前则返回一个负数,如果两个参数相等则返回0,如果第一个参数应该位于第二个参数之后则返回一个正数

function compare(value1,value2){
    if(value1 < value2){
        return -1;
    }else if(value1 > value2){
        return 1;
    }else{
        return 0;
    }
}
var array = [&#39;5px&#39;,50,1,10];
//当数字与字符串比较大小时,字符串&#39;5px&#39;会被转换成NaN,这样结果就是false
console.log(array.sort(compare));//["5px",1, 10, 50]
ログイン後にコピー

对于数值类型或valueOf()方法会返回数值类型的对象类型,比较函数可以简化

function compare(value1,value2){
    return value1 - value2;
}
var array = [&#39;5px&#39;,50,1,10];
console.log(array.sort(compare));//["5px",1,10,50]
var array = [5,50,1,10];
console.log(array.sort(compare));//[1,5,10,50]
ログイン後にコピー

如果对一个字符串数组执行不区分大小写的字母表排序,比较函数首先将参数转化为小写字符串再开始比较

a = [&#39;ant&#39;,&#39;Bug&#39;,&#39;cat&#39;,&#39;Dog&#39;];
a.sort();//[&#39;Bug&#39;,&#39;Dog&#39;,&#39;ant&#39;,&#39;cat&#39;];
a.sort(function(s,t){
    var a = s.toLowerCase();
    var b = t.toLowerCase();
    if(a < b)return -1;
    if(a > b)return 1;
    return 0;
});//[&#39;ant&#39;,&#39;bug&#39;,&#39;cat&#39;,&#39;dog&#39;]
ログイン後にコピー

【tips】使用sort()方法创建一个随机数组

function compare(){
    return Math.random() - 0.5;
}
var array = [1,2,3,4,5];
console.log(array.sort(compare));//[2,1,5,4,3]
ログイン後にコピー

数组拼接方法

【concat()】

concat()方法基于当前数组中的所有项创建一个新数组,先创建当前数组一个副本,然后将接收到的参数添加到这个副本的末尾,最后返回新构建的数组。所以concat()不影响原数组

如果不给concat()方法传递参数时,它只是复制当前的数组;如果参数是一个或多个数组,则该方法会将这些数组中的每一项都添加到结果数组中;如果传递的值不是数组,这些值就会被简单地添加到结果数组的末尾

var numbers = [1,2];
console.log(numbers,numbers.concat(3,4));//[1,2] [1,2,3,4]
console.log(numbers,numbers.concat([5,4,3],[3,4,5],1,2));//[1,2] [1,2,5,4,3,3,4,5,1,2]
console.log(numbers,numbers.concat(4,[5,[6,7]]));//[1,2] [1,2,4,5,[6,7]]
ログイン後にコピー

如果不提供参数,concat()方法返回当前数组的一个浅拷贝。所谓“浅拷贝”,指的是如果数组成员包括复合类型的值(比如对象),则新数组拷贝的是该值的引用

//该方法实际只复制了数组的第一维,数组第一维存放的是第二维的引用,而第二维才是实际存放他们的内容
var numbers = [1,2];
var newNumbers = numbers.concat();
console.log(numbers,newNumbers);//[1,2] [1,2]
numbers[0] = 0;
console.log(numbers,newNumbers);//[0,2] [1,2]

var numbers = [[1,2]];
var newNumbers = numbers.concat();
console.log(numbers,newNumbers);//[[1,2]] [[1,2]]
numbers[0][0] = 0;
console.log(numbers,newNumbers);//[[0,2]] [[0,2]]
ログイン後にコピー

concat()方法也可以用于将对象合并为数组,但是必须借助call()方法

var newArray = Array.prototype.concat.call({ a: 1 }, { b: 2 })
console.log(newArray);// [{ a: 1 }, { b: 2 }]
console.log(newArray[0].a);//1
ログイン後にコピー

创建子数组方法

【slice()】

slice()方法基于当前数组中的一个或多个项创建一个新数组,接受一个或两个参数,即要返回项的起始和结束位置,最后返回新数组,所以slice()不影响原数组

slice(start,end)方法需要两个参数start和end,返回这个数组中从start位置到(但不包含)end位置的一个子数组;如果end为undefined或不存在,则返回从start位置到数组结尾的所有项

如果start是负数,则start = max(length + start,0)

如果end是负数,则end = max(length + end,0)

start和end无法交换位置

如果没有参数,则返回原数组

var numbers = [1,2,3,4,5];
console.log(numbers.slice(2));//[3,4,5]
console.log(numbers.slice(2,undefined));//[3,4,5]
console.log(numbers.slice(2,3));//[3]
console.log(numbers.slice(2,1));//[]

console.log(numbers.slice(-3));//-3+5=2 -> [3,4,5]
console.log(numbers.slice(-8));//max(5 + -8,0)=0 -> [1,2,3,4,5]

console.log(numbers.slice(0,-3));//-3+5=2 -> [1,2]
console.log(numbers.slice(-2,-1));//-2+5=3;-1+5=4; -> [4]
ログイン後にコピー

如果不提供参数,slice()方法返回当前数组的一个浅拷贝

//该方法实际只复制了数组的第一维,数组第一维存放的是第二维的引用,而第二维才是实际存放他们的内容
var numbers = [1,2];
var newNumbers = numbers.slice();
console.log(numbers,newNumbers);//[1,2] [1,2]
numbers[0] = 0;
console.log(numbers,newNumbers);//[0,2] [1,2]

var numbers = [[1,2]];
var newNumbers = numbers.slice();
console.log(numbers,newNumbers);//[[1,2]] [[1,2]]
numbers[0][0] = 0;
console.log(numbers,newNumbers);//[[0,2]] [[0,2]]
ログイン後にコピー

slice()方法涉及到Number()转型函数的隐式类型转换,当start被转换为NaN时,相当于start = 0;当end被转换为NaN时(end为undefined除外),则输出空数组

var numbers = [1,2,3,4,5];
console.log(numbers.slice(NaN));//[1,2,3,4,5]
console.log(numbers.slice(0,NaN));//[]
console.log(numbers.slice(true,[3]));//[2,3]
console.log(numbers.slice(null,undefined));//[1,2,3,4,5]
console.log(numbers.slice({}));//[1,2,3,4,5]
console.log(numbers.slice(&#39;2&#39;,[5]));//[3,4,5]
ログイン後にコピー

可以使用slice()方法将类数组对象变成真正的数组

var arr = Array.prototype.slice.call(arrayLike);

Array.prototype.slice.call({ 0: &#39;a&#39;, 1: &#39;b&#39;, length: 2 })// [&#39;a&#39;, &#39;b&#39;]
Array.prototype.slice.call(document.querySelectorAll("p"));
Array.prototype.slice.call(arguments);
ログイン後にコピー

数组删改方法

【splice()】

splice()和slice()拥有非常相似的名字,但它们的功能却有本质的区别。splice()方法用于删除原数组的一部分成员,并可以在被删除的位置添加入新的数组成员,该方法会改变原数组

splice()返回一个由删除元素组成的数组,或者如果没有删除元素就返回一个空数组

splice()的第一个参数start指定了插入或删除的起始位置。如果start是负数,则start = max(length + start,0);如果start是NaN,则相当于start = 0

如果只提供一个元素,相当于将原数组在指定位置拆分成两个数组

var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice());// [1,2,3,4,5,6,7,8] []
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(4));// [1,2,3,4] [5,6,7,8]
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(-4));//-4+8=4; [1,2,3,4] [5,6,7,8]
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(-9));//max(-9+8,0)=0 [] [1,2,3,4,5,6,7,8]
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(NaN));//[] [1,2,3,4,5,6,7,8]
ログイン後にコピー

第二个参数number指定了应该从数组中删除的元素的个数。如果省略第二个参数,从起始点开始到数组结尾的所有元素都将被删除。如果number是负数或NaN或undefined,则number=0,因此不删除元素

var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(0,2));// [3,4,5,6,7,8] [1,2]
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(10,2));// [1,2,3,4,5,6,7,8] []
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(1,100));// [1] [2,3,4,5,6,7,8]
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(1,-5));//[1,2,3,4,5,6,7,8] []
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(1,NaN));//[1,2,3,4,5,6,7,8] []
var a = [1,2,3,4,5,6,7,8];
console.log(a,a.splice(1,undefined));//[1,2,3,4,5,6,7,8] []
ログイン後にコピー

如果后面还有更多的参数,则表示这些就是要被插入数组的新元素

var a = [1,2,3,4,5];
console.log(a,a.splice(2,0,&#39;a&#39;,&#39;b&#39;));//[1,2,&#39;a&#39;,&#39;b&#39;,3,4,5] []
console.log(a,a.splice(2,2,[1,2],3));//[1,2,[1,2],3,3,4,5] [&#39;a&#39;,&#39;b&#39;]
ログイン後にコピー

数组位置方法

ES5为数组实例添加了两个位置方法:indexOf()、lastIndexOf()

【indexOf()】

indexOf(search,start)方法接收search和start两个参数,返回search首次出现的位置,如果没有找到则返回-1

search参数表示要搜索的项;使用严格相等运算符(===)进行比较

var arr = [1,2,3,&#39;1&#39;,&#39;2&#39;,&#39;3&#39;];
console.log(arr.indexOf(&#39;2&#39;));//4
console.log(arr.indexOf(3));//2
console.log(arr.indexOf(0));//-1
ログイン後にコピー

start参数表示该搜索的开始位置,该方法会隐式调用Number()转型函数,将start非数字值(undefined除外)转换为数字。若忽略该参数或该参数为undefined或NaN时,start = 0

var arr = [&#39;a&#39;,&#39;b&#39;,&#39;c&#39;,&#39;d&#39;,&#39;e&#39;,&#39;a&#39;,&#39;b&#39;];
console.log(arr.indexOf(&#39;a&#39;,undefined));//0
console.log(arr.indexOf(&#39;a&#39;,NaN));//0
console.log(arr.indexOf(&#39;a&#39;,1));//5
console.log(arr.indexOf(&#39;a&#39;,true));//5
console.log(arr.indexOf(&#39;a&#39;,-1));//max(0,-1+7)=6; -1
console.log(arr.indexOf(&#39;a&#39;,-5));//max(0,-5+7)=2; 5
console.log(arr.indexOf(&#39;a&#39;,-50));//max(0,-50+7)=0; 0
ログイン後にコピー
var person = {name: &#39;Nicholas&#39;};
var people = [{name: &#39;Nicholas&#39;}];
var morePeople = [person];
alert(people.indexOf(person));//-1,因为person和people[0]虽然值相同,但是是两个引用
alert(morePeople.indexOf(person));//0,因为person和morepeople[0]是同一个引用
alert(morePeople.indexOf({name: &#39;Nicholas&#39;}));//-1,因为不是同一个引用
ログイン後にコピー

indexOf()方法兼容写法

if (typeof Array.prototype.indexOf != "function") {
  Array.prototype.indexOf = function (searchElement, fromIndex) {
    var index = -1;
    fromIndex = fromIndex * 1 || 0;
    for (var k = 0, length = this.length; k < length; k++) {
      if (k >= fromIndex && this[k] === searchElement) {
          index = k;
          break;
      }
    }
    return index;
  };
}
ログイン後にコピー

【lastIndexOf()】

与indexOf()不同,lastIndexOf()从右向左查找

lastIndexOf(search,start)方法接收search和start两个参数,返回search第一次出现的位置,如果没有找到则返回-1

search参数表示要搜索的项;使用严格相等运算符(===)进行比较

var arr = [1,2,3,&#39;1&#39;,&#39;2&#39;,&#39;3&#39;];
console.log(arr.lastIndexOf(&#39;2&#39;));//4
console.log(arr.lastIndexOf(3));//2
console.log(arr.lastIndexOf(0));//-1
ログイン後にコピー

start表示该搜索的开始位置,该方法会隐式调用Number()转型函数,将start非数字值(undefined除外)转换为数。若忽略该参数或该参数为undefined或NaN时,start = 0

与字符串的lastIndexOf()方法不同,当search方法为负数时,search = max(0,length+search)

var arr = [&#39;a&#39;,&#39;b&#39;,&#39;c&#39;,&#39;d&#39;,&#39;e&#39;,&#39;a&#39;,&#39;b&#39;];
console.log(arr.lastIndexOf(&#39;b&#39;));//6
console.log(arr.lastIndexOf(&#39;b&#39;,undefined));//-1
console.log(arr.lastIndexOf(&#39;a&#39;,undefined));//0
console.log(arr.lastIndexOf(&#39;b&#39;,NaN));//-1
console.log(arr.lastIndexOf(&#39;b&#39;,1));//1
console.log(arr.lastIndexOf(&#39;b&#39;,-1));//max(0,-1+7)=6; 6
console.log(arr.lastIndexOf(&#39;b&#39;,-5));//max(0,-5+7)=2; 1
console.log(arr.lastIndexOf(&#39;b&#39;,-50));//max(0,-50+7)=0; -1
ログイン後にコピー

【tips】返回满足条件的项的所有索引值

可以通过循环调用indexOf()或lastIndexOf()来找到所有匹配的项

function allIndexOf(array,value){
    var result = [];
    var pos = array.indexOf(value);
    if(pos === -1){
        return -1;
    }
    while(pos > -1){
        result.push(pos);
        pos = array.indexOf(value,pos+1);
    }
    return result;
}
var array = [1,2,3,3,2,1];
console.log(allIndexOf(array,1));//[0,5]
ログイン後にコピー

lastIndexOf()方法兼容写法

if (typeof Array.prototype.lastIndexOf != "function") {
  Array.prototype.lastIndexOf = function (searchElement, fromIndex) {
    var index = -1, length = this.length;
    fromIndex = fromIndex * 1 || length - 1;
    for (var k = length - 1; k > -1; k-=1) {
        if (k <= fromIndex && this[k] === searchElement) {
            index = k;
            break;
        }
    }
    return index;
  };
}
ログイン後にコピー

数组归并方法

数组归并方法包括reduce()和reduceRight()方法两种,它们使用指定的函数将数组元素进行组合,生成单个值。这在函数式编程中是常见的操作,也可以称为“注入”和“折叠”

【reduce()】

reduce()方法需要两个参数。第一个是执行化简操作的函数。化简函数的任务就是用某种方法把两个值组合或化简为一个值,并返回化简后的值

化简函数接受四个参数,分别是:

【1】初始变量,默认为数组的第一个元素值。函数第一次执行后的返回值作为函数第二次执行的初始变量,依次类推

【2】当前变量,如果指定了第二个参数,则该变量为数组的第一个元素的值,否则,为第二个元素的值

【3】当前变量对应的元素在数组中的索引(从0开始)

【4】原数组对象

化简函数的这四个参数之中,只有前两个是必须的,后两个则是可选的

values.reduce(function(prev, cur, index, array){
   //todo
});
ログイン後にコピー

reduce()方法第二个(可选)的参数是一个传递给函数的初始值

var a = [1,2,3,4,5];
var sum = a.reduce(function(x,y){return x+y},0);//数组求和
var product = a.reduce(function(x,y){return x*y},1);//数组求积
var max = a.reduce(function(x,y){return (x>y)?x:y;});//求最大值
ログイン後にコピー
[1, 2, 3, 4, 5].reduce(function(prev, cur){
    console.log(prev, cur)
    return prev+ cur;
});
// 1 2
// 3 3
// 6 4
// 10 5
//最后结果:15
ログイン後にコピー
[1, 2, 3, 4, 5].reduce(function(prev, cur){
    console.log(prev, cur);
    return prev + cur;
},0);
// 0 1
// 1 2
// 3 3
// 6 4
// 10 5
//最后结果:15
ログイン後にコピー

[注意]reduce()方法的返回结果类型和传入的初始值相同

[1, 2, 3, 4, 5].reduce(function(prev, cur){
    console.log(prev.sum, cur);
    prev.sum = prev.sum + cur;
    return prev;
},{sum:0});
//0 1
//1 2
//3 3
//6 4
//10 5
//Object {sum: 15}
ログイン後にコピー

利用reduce()方法,可以写一个数组求和的sum方法

Array.prototype.sum = function (){
    return this.reduce(function (prev, cur){
        return prev + cur;
    })
};
[3,4,5,6,10].sum();// 28
ログイン後にコピー

由于reduce方法依次处理每个元素,所以实际上还可以用它来搜索某个元素。比如,找出长度最长的数组元素

function findLongest(entries) {
  return entries.reduce(function (prev, cur) {
    return cur.length > prev.length ? cur : prev;
  }, &#39;&#39;);
}
console.log(findLongest([1,2,3,&#39;ab&#39;,4,&#39;bcd&#39;,5,6785,4]));//&#39;bcd&#39;
ログイン後にコピー

可以利用reduce()方法,实现二维数组的扁平化

var matrix = [
  [1, 2],
  [3, 4],
  [5, 6]
];
// 二维数组扁平化
var flatten = matrix.reduce(function (prev, cur) {
  return prev.concat(cur);
});
console.log(flatten); // [1, 2, 3, 4, 5, 6]
ログイン後にコピー

在空数组上,不带初始值参数调用reduce()将导致类型错误异常。如果调用它的时候只有一个值——数组只有一个元素并且没有指定初始值,或者有一个空数组并且指定一个初始值——reduce()只是简单地返回那个值而不会调用化简函数

var arr = [];
arr.reduce(function(){});//Uncaught TypeError: Reduce of empty array with no initial value

var arr = [];
arr.reduce(function(){},1);//1
ログイン後にコピー

reduce()方法兼容写法

if (typeof Array.prototype.reduce != "function") {
  Array.prototype.reduce = function (callback, initialValue ) {
     var previous = initialValue, k = 0, length = this.length;
     if (typeof initialValue === "undefined") {
        previous = this[0];
        k = 1;
     }
    if (typeof callback === "function") {
      for (k; k < length; k++) {
         this.hasOwnProperty(k) && (previous = callback(previous, this[k], k, this));
      }
    }
    return previous;
  };
}
ログイン後にコピー

【reduceRight()】

reduceRight()的工作原理和reduce()一样,不同的是它按照数组索引从高到低(从右到左)处理数组,而不是从低到高

var values = [1,2,3,4,5];
var sum = values.reduceRight(function(prev, cur, index, array){
    console.log(prev,cur);
    return prev + cur;
});
console.log(sum);
//5 4
//9 3
//12 2
//14 1
//15
ログイン後にコピー

reduceRight()方法兼容写法

if (typeof Array.prototype.reduceRight != "function") {
  Array.prototype.reduceRight = function (callback, initialValue ) {
    var length = this.length, k = length - 1, previous = initialValue;
    if (typeof initialValue === "undefined") {
        previous = this[length - 1];
        k--;
    }
    if (typeof callback === "function") {
       for (k; k > -1; k-=1) {          
          this.hasOwnProperty(k) && (previous = callback(previous, this[k], k, this));
       }
    }
    return previous;
  };
}
ログイン後にコピー

数组迭代方法

ECMAScript5为数组定义了5个迭代方法。每个方法都接收两个参数:要在每一项上运行的函数和(可选的)运行该函数的作用域对象——影响this的值。传入这些方法中的函数会接收三个参数:数组项的值、该项在数组中的位置和数组对象本身。根据使用的方法不同,这个函数执行后的返回值可能会也可能不会影响访问的返回值

function(item,index,array){
    //todo
}
ログイン後にコピー

【map()】

map()方法对数组的每一项运行给定函数,返回每次函数调用的结果组成的数组

//f是array的每一个元素调用的函数。它的返回值成为返回数组的元素;o是f调用时的可选this值
array.map(f,o);
ログイン後にコピー
[1,2,3].map(function(item,index,arr){return item*item});//[1,4,9]
[1,2,3].map(function(item,index,arr){return item*index});//[0,2,6]
ログイン後にコピー

map()方法还可以接受第二个参数,表示回调函数执行时this所指向的对象

var arr = [&#39;a&#39;,&#39;b&#39;,&#39;c&#39;];
[1,2].map(function(item,index,arr){return this[item]},arr);//[&#39;b&#39;,&#39;c&#39;]
ログイン後にコピー

在实际使用的时候,可以利用map()方法方便获得对象数组中的特定属性值

var users = [{name:&#39;t1&#39;,email:&#39;t1@qq.com&#39;},{name:&#39;t2&#39;,email:&#39;t2@qq.com&#39;},{name:&#39;t3&#39;,email:&#39;t3@qq.com&#39;}];
console.log(users.map(function(item,index,arr){return item.email}));//["t1@qq.com", "t2@qq.com", "t3@qq.com"]
ログイン後にコピー

map()方法还可以用于类数组对象

Array.prototype.map.call(&#39;abc&#39;,function(item,index,arr){return item.toUpperCase()});//["A", "B", "C"]
ログイン後にコピー

对于稀疏数组,map()方法不会在实际上不存在元素的序号上调用函数

var a = [1,,3];
console.log(a.map(function(item,index,arr){return item*2;}));//[2, 2: 6]
ログイン後にコピー

map()方法兼容写法

if (typeof Array.prototype.map != "function") {
  Array.prototype.map = function (fn, context) {
    var arr = [];
    if (typeof fn === "function") {
      for (var k = 0, length = this.length; k < length; k++) {      
         arr.push(fn.call(context, this[k], k, this));
      }
    }
    return arr;
  };
}
ログイン後にコピー

forEach()

forEach()方法对数组中的每一项运行给定函数,这个方法没有返回值。本质上与for循环迭代数组一样。如果需要有返回值,一般使用map方法

[1,2,3,4].forEach(function(item,index,arr){
    console.log(item)
});
//1
//2
//3
//4
ログイン後にコピー

类似于如下的for循环

var array = [1, 2, 3, 4];
for (var k = 0, length = array.length; k < length; k++) {
    console.log(array[k]);
}
ログイン後にコピー

使用forEach()方法实现简单的加法

var sum = 0;
[1, 2, 3, 4].forEach(function (item, index, array) {
    sum += item;
});
console.log(sum);//10
ログイン後にコピー

forEach()方法除了接受一个必须的回调函数参数,第二个参数还可以接受一个可选的上下文参数(改变回调函数里面的this指向)

var out = [];
[1, 2, 3].forEach(function(elem){
  this.push(elem * elem);
}, out);
console.log(out);// [1, 4, 9]
ログイン後にコピー

第二个参数对于多层this非常有用,因为多层this通常指向是不一致的,可以使用forEach()方法的第二个参数固定this

var obj = {
  name: &#39;张三&#39;,
  times: [1, 2, 3],
  print: function () {
  //该this指向obj
      console.log(this);
    this.times.forEach(function (n) {
    //该this指向window
      console.log(this);
    });
  }
};
obj.print();
ログイン後にコピー
var obj = {
  name: &#39;张三&#39;,
  times: [1, 2, 3],
  print: function () {
  //该this指向obj
      console.log(this);
    this.times.forEach(function (n) {
    //该this同样指向obj
      console.log(this);
    },this);
  }
};
obj.print();
ログイン後にコピー

forEach()循环可以用于类数组对象

var str = &#39;abc&#39;;
Array.prototype.forEach.call(str, function(item, index, array) {
  console.log( item + &#39;:&#39; + index);
});
//a:0
//b:1
//c:2
ログイン後にコピー

与for循环不同,对于稀疏数组,forEach()方法不会在实际上不存在元素的序号上调用函数

var a = [1,2,3];
delete a[1];
for(var i = 0; i < a.length; i++){
    console.log(a[i]);
}
//1
//undefined
//3
ログイン後にコピー
a.forEach(function(item,index,arr){console.log(item)});
//1
//3
ログイン後にコピー

forEach()方法无法在所有元素都传递给调用的函数之前终止遍历。也就是说,没有像for循环中使用的相应的break语句。如果要提前终止,必须把forEach()方法放在一个try块中,并能抛出一个异常

for(var i = 0; i < 5; i++){
    if(i == 2) break;
}
console.log(i);//2
ログイン後にコピー
var a = [1,2,3,4,5];
console.log(a.forEach(function(item,index,arr){
    if(index == 2) break;//Uncaught SyntaxError: Illegal break statement
}));
ログイン後にコピー
var a = [1,2,3,4,5];
a.forEach(function(item,index,arr){
    try{
      if(item == 2) throw new Error;    
    }catch(e){
        console.log(item);
    }
});
ログイン後にコピー

forEach()方法兼容写法

if(typeof Array.prototype.forEach != &#39;function&#39;){
    Array.prototype.forEach = function(fn,context){
        for(var k = 0,length = this.length; k < length; k++){
            if(typeof fn === &#39;function&#39; && Object.prototype.hasOwnProperty.call(this,k)){
                fn.call(context,this[k],k,this);
            }
        }
    }
}
ログイン後にコピー

【filter()】

filter()方法对数组中的每一项运行给定函数,返回该函数会返回true的项组成的数组。该方法常用于查询符合条件的所有数组项

[1, 2, 3, 4, 5].filter(function (elem) {
  return (elem > 3);
});// [4, 5]    

[0, 1, &#39;a&#39;, false].filter(Boolean);// [1, "a"]

[1, 2, 3, 4, 5].filter(function (elem, index, arr) {
  return index % 2 === 0;
});// [1, 3, 5]
ログイン後にコピー

filter()方法还可以接受第二个参数,指定测试函数所在的上下文对象(this对象)

var Obj = function () {
  this.MAX = 3;
};
var myFilter = function (item) {
  if (item > this.MAX) {
    return true;
  }
};
var arr = [2, 8, 3, 4, 1, 3, 2, 9];
arr.filter(myFilter, new Obj());// [8, 4, 9]
ログイン後にコピー

filter()会跳过稀疏数组中缺少的元素,它的返回数组总是稠密的,所以可以压缩稀疏数组的空缺

var a = [1,2,,,,3,,,,4];
console.log(a.length);//10
var dense = a.filter(function(){return true;})
console.log(dense,dense.length);//[1,2,3,4] 4
ログイン後にコピー

如果要压缩空缺并删除undefined和null元素,可以这样使用filter()方法

var a = [1,2,,undefined,,3,,null,,4];
console.log(a.length);//10
var dense = a.filter(function(item){return item!= undefined;})
console.log(dense,dense.length);//[1,2,3,4] 4
ログイン後にコピー

filter()方法兼容写法

if (typeof Array.prototype.filter != "function") {
  Array.prototype.filter = function (fn, context) {
    var arr = [];
    if (typeof fn === "function") {
       for (var k = 0, length = this.length; k < length; k++) {
          fn.call(context, this[k], k, this) && arr.push(this[k]);
       }
    }
    return arr;
  };
}
ログイン後にコピー

【some()】

some()方法对数组中的每一项运行给定函数,如果该函数对任一项返回true,则返回true。并且当且仅当数值中的所有元素调用判定函数都返回false,它才返回false

a = [1,2,3,4,5];
a.some(function(elem, index, arr){return elem%2===0;})//true
a.some(isNaN);//false
ログイン後にコピー

在空数组上调用some()方法会返回false

[].some(function(){});//false
ログイン後にコピー

some()方法兼容写法

if (typeof Array.prototype.some != "function") {
  Array.prototype.some = function (fn, context) {
    var passed = false;
    if (typeof fn === "function") {
         for (var k = 0, length = this.length; k < length; k++) {
          if (passed === true) break;
          passed = !!fn.call(context, this[k], k, this);
      }
    }
    return passed;
  };
}
ログイン後にコピー

【every()】

every()方法对数组中的每一项运行给定函数,如果该函数对每一项都返回true,则返回true;只要有一项返回false,则返回false

a = [1,2,3,4,5];
a.every(function(elem, index, arr){elem < 10;})//true
a.every(function(elem, index, arr){return elem%2 ===0;});//false
ログイン後にコピー

在空数组上调用every()方法会返回true

[].every(function(){});//true
ログイン後にコピー

every()方法兼容写法

if (typeof Array.prototype.every != "function") {
  Array.prototype.every = function (fn, context) {
    var passed = true;
    if (typeof fn === "function") {
       for (var k = 0, length = this.length; k < length; k++) {
          if (passed === false) break;
          passed = !!fn.call(context, this[k], k, this);
      }
    }
    return passed;
  };
}
ログイン後にコピー

总结

javascript数组方法特意定义为通用的,因此它们不仅应用在真正的数组而且在类数组对象上都能正确工作。这22种方法中,除了toString()和toLocaleString()以外的所有方法都是通用的

可以改变原数组的方法总共有7种:包括unshift()、shift()、push()、pop()这4种栈和队列方法,reverse()和sort()这2种数组排列方法,数组删改方法splice()

以上就是详细介绍JavaScript数组中的22个常用方法的代码详细介绍的内容,更多相关内容请关注PHP中文网(www.php.cn)!



関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!