国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

JavaScript數組詳解

JouyPub / 3485人閱讀

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

JavaScript數組的應用應該都比較熟悉了。

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

引用塊內容

? slice,splice
? indexOf,lastIndexOf
? sort
? 類數組對象

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

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

[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]

上面已經清晰地展現了遍歷的結果,第一列是value,第二列是對應的index值,第三列是數組本身。
現在用forEach實現數組求和:

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

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

現在看看第二個參數thisArgs的作用,如果不指定該參數,回調函數內的this指向的是window(關于this可以參照這里),例如上例中的回調函數里,你可以寫成this.price += value;,效果是一樣的(當然前提是變量確實是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);

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

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

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

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

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

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

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

//過濾出不超過10的正數
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的函數聲明:[].some( function(value, index, array) { … }, [thisArg] );
every的函數聲明:[].every( function(value, index, array) { … }, [thisArg] );
參照MDN。其實都和上面的forEach一樣,不贅述。唯一需要注意的的是回調函數需要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
兩者都是用于迭代運算。區別是reduce從頭開始迭代,reduceRight從尾開始迭代。
reduce的函數聲明:[].reduce( function(previousValue, currentValue, currentIndex, array) { … }, [initialValue] );
第一個參數是回調函數,有4個參數:previousValue,currentValue,currentIndex,array。看名字也能知道意思:前一個值,當前值,當前索引,數組本身。
第二個參數initialValue可選,表示初始值。如果省略,初始值為數組的第一個元素,這樣的話回調函數里previousValue就是第一個元素,currentValue是第二個元素。因此不設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

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

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用于復制數組,復制完后舊數組不變,返回得到的新數組是舊數組的子集。函數聲明:[].slice(begin, [end])。參照MDN
第一個參數begin是開始復制的位置,需要注意的是,可以設負數。設負數表示從尾往前數幾個位置開始復制。例如slice(-2)將從倒數第2個元素開始復制。另外需要注意的是,該參數雖未標注為可選,但實際上是可以省略的,省略的話默認為0。
第二個參數end可選,表示復制到該位置的前一個元素。例如slice(0,3)將得到前3個元素,但不包含第4個元素。不設的話默認復制到數組尾,即等于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"]

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

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"]

一個常見的應用就是刪除數組內某元素,用delete的話會留下空洞,應該用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
兩者都用于返回項目的索引值。區別是indexOf從頭開始找,lastIndexOf從尾開始找。如果查找失敗,無匹配,返回-1
indexOf的函數聲明:[].indexOf( searchElement, [fromIndex = 0] );。參照MDN
lastIndexOf的函數聲明:[].lastIndexOf( searchElement, [fromIndex = arr.length – 1] );
第一個參數searchElement即需要查找的元素。第二個參數fromIndex可選,指定開始查找的位置。如果忽略,indexOf默認是0,lastIndexOf默認是數組尾。

["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用于排序數組,函數聲明:[].sort( [sortfunction] );。參照MDN
它就一個參數,就是排序函數指針。而且是可選的,不設的話有默認的排序函數,數字的話會升序排列,string會根據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"]

但是內置的默認排序函數,是不可靠的,如下:

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

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

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]

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

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}]

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

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

JS里有很多類數組對象。什么叫類數組對象呢?它們首先是對象,并沒有繼承Array,但長的卻很像數組。最典型的如arguments對象,HTMLCollection對象。
類數組對象不能直接使用數組方法,但數組方法是如此簡單便利,要在類數組對象身上使用數組方法,需要讓數組函數通過call綁定類數組對象。
處理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 || "空"));

});

//下面這樣直接調用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會檢查參數的[[Class]]屬性,只有參數是一個真實的數組才會將數組內容連接起來,否則將作為單個元素來連接。要完全實現連接,我們需要自己在對象上增加slice方法:

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

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

更多資源上:去轉盤;或者加我的QQ群一起討論學習js,css等技術(QQ群:512245829)

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/80163.html

相關文章

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

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

    madthumb 評論0 收藏0
  • 詳解js面向對象編程

    摘要:看下面一個例子優點使用構造器函數的好處在于,它可以在創建對象時接收一些參數。按照慣例,構造函數的函數名應始終以一個大寫字母開頭,以區分普通函數。返回該對象的源代碼。使您有能力向對象添加屬性和方法。 基本概念 ECMA關于對象的定義是:無序屬性的集合,其屬性可以包含基本值、對象或者函數。對象的每個屬性或方法都有一個名字,而每個名字都映射到一個值。 類 在現實生活中,相似的對象之間往往都有...

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

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

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

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

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

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

    zhaofeihao 評論0 收藏0

發表評論

0條評論

JouyPub

|高級講師

TA的文章

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