對(duì)對(duì)象的理解和探索第喳,可以從console.log()
或console.dir()
看得很詳細(xì)满钟,一個(gè)簡(jiǎn)單示例
var a = [1,2,3]
var b = {x:1,y:2,z:3}
var c = new String(123)
console.log(a,b,c)
來看一個(gè)例子:首先定義一個(gè)對(duì)象眯勾,給對(duì)象添加數(shù)組的
push
和splice
方法,然后執(zhí)行一下push
方法婆誓。
var obj={
'2':3,
'3':4,
'length':2,
'splice':Array.prototype.splice,
'push':Array.prototype.push
}
obj.push(1)
obj.push(2)
console.log(obj)
這似乎是不科學(xué)的吃环,對(duì)象怎么能使用數(shù)組的方法呢?我們先來看一下執(zhí)行結(jié)果
2洋幻、3
的值和length
全部改變郁轻,證明塞進(jìn)去的方法起效了。我們先看一下
Array?.prototype?.push()
文留,push()
方法將一個(gè)或多個(gè)元素添加到數(shù)組的末尾好唯,并返回該數(shù)組的新長(zhǎng)度。由此可見該方法是有length
返回值的燥翅,似乎都對(duì)應(yīng)上了骑篙,那么到底是怎么實(shí)現(xiàn)的呢?
//push的內(nèi)部原理
Arrary.prototype.push = function(target){
this[this.length] = target //把目標(biāo)值設(shè)為this數(shù)組的最后一項(xiàng)
this.length ++ //設(shè)置數(shù)組的長(zhǎng)度权旷,使其+1
}
//----------------------
var a =[1,2,3]
console.log(a.length) //---> 3
console.log(a[a.length-1]) //--->3
//我們套入剛才的對(duì)象進(jìn)行比對(duì)
var obj={
'2':3,
'3':4,
'length':2,
'splice':Array.prototype.splice,
'push':Array.prototype.push
}
obj.push(1)
//-------------------------------
obj.push(1) = 1 => {
obj[obj.length] = 1 //obj[2] = 1
obj.length ++ //obj.length = 3
}
//-------------------------------
obj.push(2)
//-------------------------------
obj.push(2) = 2 => {
obj[obj.length] = 2 //obj[3] = 2
obj.length ++ //obj.length = 4
}
//-------------------------------
console.log(obj)
/*所以obj里的值就出來了
obj = {
2: 1
3: 2
length: 4
}
吃透了原理替蛉,原來JS像加減法一樣簡(jiǎn)單*/
JS的世界如此奇妙,食髓知味拄氯,那么我們?cè)賮硌芯恳幌?code>splice()的原理吧躲查。
Array?.prototype?.splice()
,splice()
方法通過刪除或替換現(xiàn)有元素或者原地添加新的元素來修改數(shù)組译柏,并以數(shù)組形式返回被修改的內(nèi)容镣煮。此方法會(huì)改變?cè)瓟?shù)組。
簡(jiǎn)而言之鄙麦,array.splice(start,deleteCount,value,...)
方法實(shí)現(xiàn)了三個(gè)功能:插入典唇,刪除和替換,而且刪除了元素之后還會(huì)返回一個(gè)包含刪除元素的新數(shù)組
var a = [1,2,3,4]
console.log(a.splice(1,2),a)
//[2,3],[1,4] 在索引值為1的位置刪除了兩個(gè)元素(刪除)
console.log(a.splice(1,0,2,3),a)
//[],[1,2,3,4] 在索引值為1的位置刪除了0個(gè)元素并添加了兩個(gè)元素(插入)
console.log(a.splice(1,2,5,6),a)
//[2,3],[1,5,6,4] 在索引值為1的位置刪除了兩個(gè)元素并添加了兩個(gè)元素(替換)
我們分析另外幾種情況:array.splice(s,d,v,...)
- 刪除
(s,d)
①s
為-1胯府,代表從倒數(shù)第1個(gè)開始介衔;②d
不傳參,代表從s
處刪到尾巴骂因;③d
為負(fù)數(shù)炎咖,需要傳遞v
,0或負(fù)數(shù)代表插入元素。 - 插入
(s,0,v,...)
在s
處插入v.length
個(gè)元素乘盼,后段元素后移升熊。 - 替換
(s,d,v,...)
在s
處刪除d
個(gè)元素,插入v.length
個(gè)元素绸栅,視d
和v.length
大小操作后段元素前移或后移级野。
splice()
方法一直詬病較多,從幾種方法的實(shí)現(xiàn)不難看出粹胯,start
及之后的元素被全部重新排列了蓖柔,很是浪費(fèi)時(shí)間,我們先嘗試這幾種操作的代碼實(shí)現(xiàn)
Array?.prototype?.splice(start,deleteCount,value) = function(s,d,v){
for(var i = 0,i<d,i++){ this[s] } 新建數(shù)組风纠,根據(jù)s d傳入刪除元素渊抽,等待返回;調(diào)用內(nèi)部方法刪除指定位置數(shù)組元素议忽;準(zhǔn)備value列表,在s處插入十减,其后元素依次向后平移栈幸;考慮d>v.length時(shí),原數(shù)組后段元素向前平移……}阿西吧帮辟,好吧速址,我說著玩的,嘗試失敗【手動(dòng)捂臉】
回到前面的例子由驹,推導(dǎo)不出來詳細(xì)的實(shí)現(xiàn)代碼怎么計(jì)算所得值呢芍锚?上邊測(cè)試結(jié)果用的是火狐開發(fā)者工具,現(xiàn)在轉(zhuǎn)用chrome蔓榄,截張圖看看
var obj={
'2':3,
'3':4,
'length':4,
'splice':Array.prototype.splice,
'push':Array.prototype.push
}
console.log(obj)
Object(4) [empty × 2, 3, 4, splice: ?, push: ?]
看到這里就好玩了并炮,chrome把符合數(shù)組結(jié)構(gòu)的對(duì)象(有數(shù)組的屬性)注釋為數(shù)組了,而對(duì)應(yīng)的鍵變成了索引般的存在甥郑。這里把length
變成了4逃魄,是為了對(duì)應(yīng)“索引”3(3:4),這樣再按數(shù)組計(jì)算就可以了澜搅。
obj.splice(2,1)
操作[empty,empty,3,4]
數(shù)組伍俘,應(yīng)該得到[empty,empty,4]
,再回到原對(duì)象結(jié)構(gòu)勉躺,就是{2:4,length:3}
我們截個(gè)圖see see
我們來走個(gè)測(cè)試癌瘾,驗(yàn)證一下
var obj={
'2':3,
'3':4,
'length':4,
'splice':Array.prototype.splice,
'push':Array.prototype.push
} //[empty,empty,3,4]
obj.splice(2,0,5,6) //[empty,empty,5,6,3,4]
obj.splice(0,4,1,2) //[1,2,3,4]
console.log(obj)
當(dāng)對(duì)象擁有數(shù)組的“l(fā)ength”和“索引”時(shí)饵溅,可以認(rèn)為是個(gè)偽數(shù)組
[ArrayLike]
妨退,我們這個(gè)等于強(qiáng)塞進(jìn)去兩個(gè)數(shù)組的方法,需要注意的是,push()
和splice()
還不太一樣碧注,push()
是特意設(shè)計(jì)為通用的嚣伐,splice()
對(duì)對(duì)象用起來還有局限,start
超過length
的值就操作不動(dòng)了萍丐。再看一眼官方push()
的例子
var obj = {
length: 0,
addElem: function addElem (elem) {
[].push.call(this, elem);
}
};
//加兩個(gè)空對(duì)象
obj.addElem({});
obj.addElem({});
console.log(obj.length,obj);
// → 2 {0: {}, 1: {}, length: 2, addElem: ?}
異曲同工轩端,不過人家的更為規(guī)整。本篇先記錄到這逝变,發(fā)現(xiàn)好玩的例子再來添加基茵。