成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

JavaScript數(shù)組詳解

JouyPub / 3487人閱讀

摘要:唯一需要注意的的是回調(diào)函數(shù)需要有值,否則新數(shù)組都是。唯一需要注意的的是回調(diào)函數(shù)需要布爾值或,如果忘記寫語句,返回得到的是空數(shù)組,表示一個都不匹配。

JavaScript數(shù)組的應(yīng)用應(yīng)該都比較熟悉了。

? forEach,map,filter
? some,every
? reduce,reduceRight

引用塊內(nèi)容

? slice,splice
? indexOf,lastIndexOf
? sort
? 類數(shù)組對象

forEach,map,filter
forEach遍歷數(shù)組,函數(shù)聲明:[].forEach( function(value, index, array) { … }, [thisArg] );。

第一個參數(shù)是回調(diào)函數(shù),它支持3個參數(shù),第1個是遍歷的數(shù)組內(nèi)容,第2個是對應(yīng)索引,第3個是數(shù)組自身。
第二個參數(shù)thisArg可選,可用于以改變回調(diào)函數(shù)里面的this指針
因為forEach是第一個被介紹的數(shù)組方法,所以稍微詳細(xì)點用console.log看一下回調(diào)函數(shù)的3個參數(shù)。(之后的數(shù)組方法有興趣的可以自己用console.log看一下回調(diào)函數(shù),不贅述)

[1, 2 ,3, 4].forEach(console.log);
// 1, 0, [1, 2, 3, 4]
// 2, 1, [1, 2, 3, 4]
// 3, 2, [1, 2, 3, 4]
// 4, 3, [1, 2, 3, 4]

上面已經(jīng)清晰地展現(xiàn)了遍歷的結(jié)果,第一列是value,第二列是對應(yīng)的index值,第三列是數(shù)組本身。
現(xiàn)在用forEach實現(xiàn)數(shù)組求和:

var price = 0;
[1, 2, 3, 4].forEach(function (value) {
    price += value;
});
console.log(price);     //10

相比for循環(huán),上述代碼除了更簡單外,還避免了常見的for循環(huán)的起始,終止條件越界等錯誤。對于數(shù)組遍歷來說,forEach和map是優(yōu)于for循環(huán)的。

現(xiàn)在看看第二個參數(shù)thisArgs的作用,如果不指定該參數(shù),回調(diào)函數(shù)內(nèi)的this指向的是window(關(guān)于this可以參照這里),例如上例中的回調(diào)函數(shù)里,你可以寫成this.price += value;,效果是一樣的(當(dāng)然前提是變量確實是window的全局屬性)。但有時this指向window就不對了,如下:

var group = {
    members: ["Jack", "Andy", "Natasha"],
    joinParty: "Yes",
    getInfo: function (m) {
        this.isJoinParty(m);
        console.log(m + " " + this.joinParty);
    },
    isJoinParty: function (m) {
        switch(m) {
        case "Andy" :
            this.joinParty = "No";
            break;
        default:
            this.joinParty = "Yes";
            break;
        }
    }
};
group.members.forEach(group.getInfo);

代碼很簡單,小組內(nèi)3人,Andy不參加聚會,另兩人參加聚會。期望把統(tǒng)計結(jié)果打印出來。但很遺憾上面代碼會報Error。按理說getInfo函數(shù)里的this應(yīng)該指向group對象,但遺憾地是getInfo作為[].forEach的回調(diào)函數(shù)時相當(dāng)于普通函數(shù),因此getInfo里的this指向的是window。而window對象里顯然不存在isJoinParty。
因此正確的調(diào)用方式是,添加第二個參數(shù),明確指定this的綁定對象:
group.members.forEach(group.getInfo, group);
//Jack Yes
//Andy No
//Natasha Yes

map映射創(chuàng)建新數(shù)組,函數(shù)聲明:[].map( function(value, index, array) { … }, [thisArg] );。和forEach一樣,不贅述。唯一需要注意的的是回調(diào)函數(shù)需要有return值,否則新數(shù)組都是undefined。

其實map能干的事forEach都能干,你可以把map理解為forEach的一個特例,專門用于:通過現(xiàn)有的數(shù)組建立新數(shù)組。例如將舊數(shù)組中字符串都trim一下,去除空格后生成新數(shù)組:

var trimmed = ["    Jack","Betty    ","    Chirs    "].map(function(s) {
    return s.trim();    //需要return值,否則新數(shù)組里都是undefined
});
console.log(trimmed);    //["Jack", "Betty", "Chirs"]

在沒有map之前是通過forEach來創(chuàng)建新數(shù)組的。你需要先定義一個空數(shù)組,再將每次trim后的字符串push到新數(shù)組內(nèi),比較麻煩。因為“通過現(xiàn)有的數(shù)組建立新數(shù)組”這個需求是如此的普遍,因此ES5中干脆追加了map方法,相比forEach代碼簡單優(yōu)雅多了。

filter用于過濾數(shù)組,函數(shù)聲明:[].filter( function(value, index, array) { … }, [thisArg] );。和forEach一樣,不贅述。唯一需要注意的的是回調(diào)函數(shù)需要return布爾值true或false,如果忘記寫return語句,返回得到的是空數(shù)組,表示一個都不匹配。例如:

var newArray = [0, 1, 2].filter(function(value) {});
console.log(newArray);         //[],沒有return語句得到的是空數(shù)組

//過濾出不超過10的正數(shù)
var newArray2 = [0, 1, 2, 14].filter(function(value) {
    return value > 0 && value <= 10;
});
console.log(newArray2);   //[1, 2]

some,every
some表示只要某一個滿足條件就OK,every表示全部滿足條件才OK。
some的函數(shù)聲明:[].some( function(value, index, array) { … }, [thisArg] );
every的函數(shù)聲明:[].every( function(value, index, array) { … }, [thisArg] );
參照MDN。其實都和上面的forEach一樣,不贅述。唯一需要注意的的是回調(diào)函數(shù)需要return布爾值true或false,如果忘記寫return語句,表示不滿足條件,返回false

[1, 10, 100].some(function(x) { x > 5; });         //false,忘記寫return了
[1, 2, 3, 4, 5].every(function(x) { x > 0; });     //false,忘記寫return了

[1, 10, 100].some(function(x) { return x > 5; });     // true
[1, 10, 100].some(function(x) { return x < 0; });     // false 
[1, 2, 3, 4, 5].every(function(x) { return x > 0; });     // true
[1, 2, 3, 4, 5].every(function(x) { return x < 3; });     // false

reduce,reduceRight
兩者都是用于迭代運算。區(qū)別是reduce從頭開始迭代,reduceRight從尾開始迭代。
reduce的函數(shù)聲明:[].reduce( function(previousValue, currentValue, currentIndex, array) { … }, [initialValue] );
第一個參數(shù)是回調(diào)函數(shù),有4個參數(shù):previousValue,currentValue,currentIndex,array。看名字也能知道意思:前一個值,當(dāng)前值,當(dāng)前索引,數(shù)組本身。
第二個參數(shù)initialValue可選,表示初始值。如果省略,初始值為數(shù)組的第一個元素,這樣的話回調(diào)函數(shù)里previousValue就是第一個元素,currentValue是第二個元素。因此不設(shè)initialValue的話,會少一次迭代。例如:

var sum = [1, 2, 3, 4].reduce(function (previous, current) {
    return previous + current;
});
console.log(sum);    //10

//給它加上initialValue初始值10
var sum2 = [1, 2, 3, 4].reduce(function (previous, current) {
    return previous + current;
}, 10);
console.log(sum2);    //20

上圖清楚地表明了各個運算步驟,很容易理解。如果不設(shè)initialValue,會少一次迭代。
reduceRight的函數(shù)聲明:[].reduceRight( function(previousValue, currentValue, currentIndex, array) { … }, [initialValue] );。和reduce一樣,不贅述
用reduce和reduceRight很容易就能實現(xiàn)二維數(shù)組扁平化,如下:

var flat1 = [[0, 1], [2, 3], [4, 5]].reduce(function(a, b) {
    return a.concat(b);
});
console.log(flat1);    //[0, 1, 2, 3, 4, 5]

var flat2 = [[0, 1], [2, 3], [4, 5]].reduceRight(function(a, b) {
    return a.concat(b);
});
console.log(flat2);    //[4, 5, 2, 3, 0, 1]

slice,splice
兩者做的事情還不太一樣,但名字實在太像了,所以放一起介紹。
slice用于復(fù)制數(shù)組,復(fù)制完后舊數(shù)組不變,返回得到的新數(shù)組是舊數(shù)組的子集。函數(shù)聲明:[].slice(begin, [end])。參照MDN
第一個參數(shù)begin是開始復(fù)制的位置,需要注意的是,可以設(shè)負(fù)數(shù)。設(shè)負(fù)數(shù)表示從尾往前數(shù)幾個位置開始復(fù)制。例如slice(-2)將從倒數(shù)第2個元素開始復(fù)制。另外需要注意的是,該參數(shù)雖未標(biāo)注為可選,但實際上是可以省略的,省略的話默認(rèn)為0。
第二個參數(shù)end可選,表示復(fù)制到該位置的前一個元素。例如slice(0,3)將得到前3個元素,但不包含第4個元素。不設(shè)的話默認(rèn)復(fù)制到數(shù)組尾,即等于array.length。

var fruits = ["Banana", "Orange", "Lemon", "Apple", "Mango"].slice(0, 3);
console.log(fruits);    //["Banana", "Orange", "Lemon"]

var fruits2 = ["Banana", "Orange", "Lemon", "Apple", "Mango"].slice(-1);
console.log(fruits2);    //["Mango"]

當(dāng)然slice最常見的是用在將類數(shù)組arguments對象轉(zhuǎn)換為真正的數(shù)組:
var args = [].slice.call(arguments);
splice用于剝離數(shù)組,從舊數(shù)組中移除元素,返回得到的新數(shù)組是被移除的元素。函數(shù)聲明:[].splice(start, deleteCount, [item…])。參照MDN
第一個參數(shù)start是開始剝離的位置,需要注意的是,可以設(shè)負(fù)數(shù)。設(shè)負(fù)數(shù)表示從尾往前數(shù)幾個位置開始剝離。例如splice (-2)將從倒數(shù)第2個元素開始剝離。
第二個參數(shù)deleteCount是要剝離的元素個數(shù),設(shè)0表示一個都不剝離。
第三個參數(shù)開始可選,用于替換舊數(shù)組中被移除的元素

var oldArray = ["a", "b", "c"];
var newArray = oldArray.splice(1, 2, "Jack", "Betty", "Andy");
console.log(oldArray);     //["a", "Jack", "Betty", "Andy"]
console.log(newArray);     //["b", "c"]

一個常見的應(yīng)用就是刪除數(shù)組內(nèi)某元素,用delete的話會留下空洞,應(yīng)該用splice方法:
//錯誤的方法用delete

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

//正確的方法用splice
numbers.splice(2, 1); 
console.log(numbers);     //[0, 1, 3, 4]

indexOf,lastIndexOf
兩者都用于返回項目的索引值。區(qū)別是indexOf從頭開始找,lastIndexOf從尾開始找。如果查找失敗,無匹配,返回-1
indexOf的函數(shù)聲明:[].indexOf( searchElement, [fromIndex = 0] );。參照MDN
lastIndexOf的函數(shù)聲明:[].lastIndexOf( searchElement, [fromIndex = arr.length – 1] );
第一個參數(shù)searchElement即需要查找的元素。第二個參數(shù)fromIndex可選,指定開始查找的位置。如果忽略,indexOf默認(rèn)是0,lastIndexOf默認(rèn)是數(shù)組尾。

["a", "b", "d", "e"].indexOf("b");      //1
["a", "b", "d", "e"].indexOf("b", 2);   //-1,從2號位開始找沒找到
["a", "b", "d", "e"].indexOf("c");      //-1,沒找到

["a", "b", "d", "e"].lastIndexOf("b");      //1
["a", "b", "d", "e"].lastIndexOf("b", 2);   //1,逆向2號位等價于正向1號位
["a", "b", "d", "e"].lastIndexOf("c");      //-1,沒找到

sort
sort用于排序數(shù)組,函數(shù)聲明:[].sort( [sortfunction] );。參照MDN
它就一個參數(shù),就是排序函數(shù)指針。而且是可選的,不設(shè)的話有默認(rèn)的排序函數(shù),數(shù)字的話會升序排列,string會根據(jù)Unicode升序排列。

var sumArray = [4, 3, 1, 0, 2];
var sumArray2 = ["d", "z", "a"];
sumArray.sort();
sumArray2.sort();
console.log(sumArray);  //[0, 1, 2, 3, 4]
console.log(sumArray2); //["a", "d", "z"]

但是內(nèi)置的默認(rèn)排序函數(shù),是不可靠的,如下:

var scores = [1, 10, 2, 21]; 
scores.sort(); 
console.log(scores);    //[1, 10, 2, 21]

因此保險起見最好自定義排序函數(shù):

var scores = [1, 10, 2, 21]; 
function compareNumbers(x, y) {
    if (x < y) { return -1; } 
    if (x > y) { return 1; }
    return 0;
}
scores.sort(compareNumbers);
console.log(scores);        //[1, 2, 10, 21]

而且如果數(shù)組內(nèi)是對象,或排序邏輯復(fù)雜的話,那默認(rèn)排序函數(shù)更是力不從心了,必須自定義排序函數(shù):

var items = [
    { name: "Jack", value: 37 },
    { name: "Betty", value: 21 },
    { name: "Andy", value: 45 }
];
items.sort(function (a, b) {
    if (a.value < b.value) { return -1; } 
    if (a.value > b.value) { return 1; }
    return 0;
});
console.log(items);
//[{ name="Betty", value=21},
// { name="Jack", value=37}, 
// { name="Andy",  value=45}]

剩下的比較簡單,大致說一下,就不詳細(xì)介紹了。
push和pop用于數(shù)組尾處壓入和彈出元素。
unshift和shift用于數(shù)組頭部壓入和彈出元素。
reverse用于反轉(zhuǎn)數(shù)組,concat用于連接數(shù)組,join用于數(shù)組元素間插入些東西后拼接成string。
類數(shù)組

var arr1 = new Array();
arr1.push(1);
arr1.push(2);
arr1.push(3);
console.log(arr1); //[1,2,3]
console.log(arr1.pop()); //3 彈出棧頂數(shù)據(jù)

JS里有很多類數(shù)組對象。什么叫類數(shù)組對象呢?它們首先是對象,并沒有繼承Array,但長的卻很像數(shù)組。最典型的如arguments對象,HTMLCollection對象。
類數(shù)組對象不能直接使用數(shù)組方法,但數(shù)組方法是如此簡單便利,要在類數(shù)組對象身上使用數(shù)組方法,需要讓數(shù)組函數(shù)通過call綁定類數(shù)組對象。
處理arguments對象:
var args = [].slice.call(arguments);
處理HTMLCollection對象:
//用forEach遍歷頁面所有div,輸入className
var divs = document.getElementsByTagName("div");
Array.prototype.forEach.call(divs, function(div) {

console.log("該div類名是:" + (div.className || "空"));

});

//下面這樣直接調(diào)用forEach將報錯,因為divs是HTMLCollection對象而非Array
divs.forEach(function(div) {

console.log("該div類名是:" + (div.className || "空"));

});
處理字面量對象:

var arrayLike = { 0: "a", 1: "b", 2: "c", length: 3 };
var result = Array.prototype.map.call(arrayLike, function(s) {
    return s.toUpperCase();
});     
console.log(result);    //["A", "B", "C"]
處理字符串:
var result = Array.prototype.map.call("abc", function(s) {
    return s.toUpperCase();
});
console.log(result);    //["A", "B", "C"]

但Array的concat會檢查參數(shù)的[[Class]]屬性,只有參數(shù)是一個真實的數(shù)組才會將數(shù)組內(nèi)容連接起來,否則將作為單個元素來連接。要完全實現(xiàn)連接,我們需要自己在對象上增加slice方法:

//單用concat的話,arguments對象將作為一個單一整體被連接
function namesColumn() {
    return ["Jack"].concat(arguments);
}
var newNames = namesColumn("Betty", "Andy", "Chris");
console.log(newNames);    //["Jack", ["Betty", "Andy", "Chris"]]

//配合slice能實現(xiàn)完全連接
function namesColumn() {
    return ["Jack"].concat([].slice.call(arguments));
}
var newNames = namesColumn("Betty", "Andy", "Chris");
console.log(newNames);    //["Jack", "Betty", "Andy", "Chris"]

更多資源上:去轉(zhuǎn)盤;或者加我的QQ群一起討論學(xué)習(xí)js,css等技術(shù)(QQ群:512245829)

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/80163.html

相關(guān)文章

  • 【連載】前端個人文章整理-從基礎(chǔ)到入門

    摘要:個人前端文章整理從最開始萌生寫文章的想法,到著手開始寫,再到現(xiàn)在已經(jīng)一年的時間了,由于工作比較忙,更新緩慢,后面還是會繼更新,現(xiàn)將已經(jīng)寫好的文章整理一個目錄,方便更多的小伙伴去學(xué)習(xí)。 showImg(https://segmentfault.com/img/remote/1460000017490740?w=1920&h=1080); 個人前端文章整理 從最開始萌生寫文章的想法,到著手...

    madthumb 評論0 收藏0
  • 詳解js面向?qū)ο缶幊?/b>

    摘要:看下面一個例子優(yōu)點使用構(gòu)造器函數(shù)的好處在于,它可以在創(chuàng)建對象時接收一些參數(shù)。按照慣例,構(gòu)造函數(shù)的函數(shù)名應(yīng)始終以一個大寫字母開頭,以區(qū)分普通函數(shù)。返回該對象的源代碼。使您有能力向?qū)ο筇砑訉傩院头椒ā? 基本概念 ECMA關(guān)于對象的定義是:無序?qū)傩缘募希鋵傩钥梢园局?、對象或者函?shù)。對象的每個屬性或方法都有一個名字,而每個名字都映射到一個值。 類 在現(xiàn)實生活中,相似的對象之間往往都有...

    lolomaco 評論0 收藏0
  • javascript map()詳解

    摘要:不會對空數(shù)組進行遍歷遍歷數(shù)組的每一項,數(shù)組當(dāng)前項的下標(biāo),原數(shù)組函數(shù)內(nèi)沒有執(zhí)行,證明數(shù)組為空是并不執(zhí)行遍歷返回一個新數(shù)組,長度等于原數(shù)組長度遍歷數(shù)組的每一項,數(shù)組當(dāng)前項的下標(biāo),原數(shù)組即便函數(shù)返回空結(jié)果數(shù)組的 map() 不會對空數(shù)組進行遍歷 let arr = [] let newArr = arr.map((item, i, arr) => { ...

    suxier 評論0 收藏0
  • 常用JavaScript小技巧及原理詳解

    摘要:使用一元加模擬函數(shù)原理對非數(shù)值類型的數(shù)據(jù)使用一元加,會起到與函數(shù)相同的效果。中,若判斷不為則不再進行下一步操作。使用邏輯或設(shè)置默認(rèn)值邏輯或也屬于短路操作,即當(dāng)?shù)谝粋€操作數(shù)可以決定結(jié)果時,不再對第二個操作數(shù)進行求值。 善于利用JS中的小知識的利用,可以很簡潔的編寫代碼 1. 使用!!模擬Boolean()函數(shù) 原理:邏輯非操作一個數(shù)據(jù)對象時,會先將數(shù)據(jù)對象轉(zhuǎn)換為布爾值,然后取反,兩個!!...

    chnmagnus 評論0 收藏0
  • JavaScript 特殊對象 Array-Like Objects 詳解

    摘要:很簡單,不是數(shù)組,但是有屬性,且屬性值為非負(fù)類型即可。至于屬性的值,給出了一個上限值,其實是感謝同學(xué)指出,因為這是中能精確表示的最大數(shù)字。如何將函數(shù)的實際參數(shù)轉(zhuǎn)換成數(shù)組 這篇文章拖了有兩周,今天來跟大家聊聊 JavaScript 中一類特殊的對象 -> Array-Like Objects。 (本文節(jié)選自 underscore 源碼解讀系列文章,完整版請關(guān)注 https://githu...

    zhaofeihao 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<