JS引用類型较坛,對(duì)象拷貝

引用類型

引用類型變量保存的僅僅是一個(gè)指針,指針指向堆內(nèi)存中保存對(duì)象的位置块差。

  • 所以基本類型復(fù)制的時(shí)候僅僅復(fù)制值侵续,復(fù)制前后的兩個(gè)變量相互獨(dú)立
  • 而引用類型復(fù)制的時(shí)候?qū)嵸|(zhì)上僅僅是把保存對(duì)象的位置的地址復(fù)制傳給了復(fù)制后的變量憨闰,兩個(gè)變量實(shí)際上指向的還是同一個(gè)對(duì)象,所以通過(guò)一個(gè)變量改變對(duì)象的屬性需五,另外一個(gè)變量引用對(duì)象時(shí)鹉动,屬性也會(huì)發(fā)生變化。

對(duì)于函數(shù)的參數(shù)傳遞宏邮,不用去考慮按值傳遞還是按引用傳遞泽示,只需要這么理解:

     function fn (n) {
//         var n = a; // 可以視為函數(shù)參數(shù)的傳遞,將a中的值傳給n
         n++;
     }
     var a = 10;
     fn(a);
     console.log(a);   //10   雖然n變?yōu)?1蜜氨,但是a仍然為10

可以理解成函數(shù)內(nèi)部有一句隱式的語(yǔ)句var n = a; 可以視為函數(shù)參數(shù)的傳遞械筛,將a中的值傳給n,a是基本類型時(shí)傳遞的就是值,a是引用類型時(shí)a中保存的就是對(duì)象的地址飒炎,所以將地址傳遞給n埋哟。
當(dāng)a是引用類型時(shí)的例子

     function fn (n) {
//         var n = a; // 傳遞的是a的對(duì)象的地址
         n.age++;
     }
     var a = {
         name: 'deejay',
         sex: 'male',
         age: 21
     }
     fn(a);
     console.log(a);   // n.age =22 a.age =22,都為22,因?yàn)閭鬟f的是參數(shù)郎汪,實(shí)質(zhì)上改變的是同一個(gè)對(duì)象

對(duì)象拷貝

  • 淺拷貝:淺拷貝即僅僅拷貝一層的對(duì)象屬性赤赊。舉例說(shuō)明:
    var oldObj = {
        name: 'deejay',
        age: 20,
        friend: {
            name: 'dee',
            age: 20
        }
    }
    function shallowCopy(oldObj) {
        var newObj = {}; // 要進(jìn)行對(duì)象的拷貝,必須要新開辟一個(gè)空間存放對(duì)象煞赢,有不同的地址抛计,才能相互獨(dú)立。
        for(var key in oldObj) {
            newObj[key] = oldObj[key];//創(chuàng)建了一個(gè)新對(duì)象之后照筑,遍歷oldObj的屬性吹截,每一項(xiàng)都拷貝給newObj的屬性
        }
        return newObj;
    }
    var newObj = shallowCopy(oldObj);
    console.log(newObj);
    newObj.age++;
    console.log(oldObj.age);  //20  newobj的age增加,oldobj的age仍然不變
    newObj.friend.age++;
    console.log(oldObj.friend.age); // 21  newobj中的friend中的age增加凝危,oldobj的age也跟著一起增加了

上述例子中的拷貝對(duì)象oldObj有兩層波俄,淺拷貝只能使得拷貝后的對(duì)象newObj的第一層屬性跟oldObj相互獨(dú)立。

  • 深拷貝: 針對(duì)淺拷貝的問(wèn)題媒抠,深拷貝可以完全拷貝對(duì)象弟断,不管有多少層屬性,都可以相互獨(dú)立趴生,思路方法有:
  1. 依靠遞歸:
    基本思路如下:
    function deepCopy (oldObj) {
        var newObj = {}; //一樣阀趴,還是先創(chuàng)建個(gè)新對(duì)象
        for(var key in oldObj) {  // 遍歷oldObj中的所有屬性
            if (typeof oldObj[key] === 'number' || typeof oldObj[key] === 'string') {
                //進(jìn)行判斷昏翰,屬性值中是不是基本類型,即沒有嵌套的情況下刘急,依然按照淺拷貝的方式復(fù)制屬性到newObj中棚菊,本例中oldObj屬性只有number和string,就少寫幾種情況
                newObj[key] = oldObj[key];
            }
            else if (typeof oldObj[key] === 'Object') {  
                //當(dāng)檢測(cè)到有嵌套的情況時(shí)叔汁,使用遞歸统求,調(diào)用本身函數(shù)對(duì)嵌套的屬性進(jìn)行拷貝,如果嵌套的屬性中還有嵌套的屬性据块,那么繼續(xù)遞歸直到?jīng)]有嵌套為止码邻,拷貝所有屬性
                newObj[key] = deepCopy(oldObj[key]);
            }        
        }
         return newObj;
    }

上述代碼僅僅表述了深拷貝的思路,那么實(shí)際書寫代碼的時(shí)候另假,還需要考慮oldObj中繼承而來(lái)的屬性像屋,所以要針對(duì)我們自己定義的屬性來(lái)進(jìn)行拷貝,通過(guò)使用hasOwnProperty方法來(lái)判斷边篮,如果是我們自己定義的屬性己莺,那么再進(jìn)行拷貝,在判斷是否嵌套的條件中戈轿,typeof屬性的值還可能是'number','string','boolean',還可能是null以及undefined凌受。所以完整的深拷貝代碼如下:

    var oldObj = {
        name: 'deejay',
        age: 20,
        friend: {
            name: 'dee',
            age: 20,
        }
    }
    function deepCopy (oldObj) {
        var newObj = {}; //創(chuàng)建一個(gè)地址不同的新對(duì)象
            for (var key in oldObj) { //遍歷oldObj中的所有屬性
                if (oldObj.hasOwnProperty(key)) {  // 針對(duì)自身?yè)碛械亩x的屬性進(jìn)行拷貝
                    if (typeof oldObj[key] === 'string' || typeof oldObj[key] === 'number' || typeof oldObj[key] === 'boolean'
                            || oldObj[key] === null || oldObj[key] === undefined) { //完整的判斷是否嵌套的條件
                        newObj[key] = oldObj[key];
                    }
                    else {  // 不是上面判斷的條件,即為嵌套的屬性思杯,通過(guò)遞歸進(jìn)行拷貝
                        newObj[key] = deepCopy(oldObj[key]);
                    }
                }
            }
        return newObj;
    }
    var newObj = deepCopy(oldObj);
    console.log(newObj);
    newObj.age++;
    console.log(oldObj.age);  //20  newobj的age增加胜蛉,oldobj的age仍然不變
    newObj.friend.age++;
    console.log(oldObj.friend.age); // 20  newobj中的friend中的age增加,oldObj仍然不變智蝠,newObj和oldObj是相互獨(dú)立的
//    不管有多少層嵌套的屬性腾么,都是相互獨(dú)立的,即為深拷貝
  1. 使用JSON對(duì)象的stringify方法和parse方法:
    var oldObj = {
        name: 'deejay',
        age: 20,
        friend: {
            name: 'dee',
            age: 20,
        }
    }
    var newObj = JSON.parse(JSON.stringify(oldObj));
    /*stringify方法可以把一個(gè)對(duì)象轉(zhuǎn)換為一個(gè)字符串杈湾,然后parse方法可以將字符串轉(zhuǎn)化為一個(gè)對(duì)象解虱,這樣得到的新對(duì)象就跟原來(lái)對(duì)象完全獨(dú)立了*/
    console.log(newObj);
    newObj.age = 25;
    console.log(oldObj.age);  // 20   newObj和oldObj完全獨(dú)立,不受影響
    newObj.friend.age = 25;
    console.log(oldObj.friend.age); //20 newObj和oldObj完全獨(dú)立漆撞,不受影響

1.引用類型有哪些殴泰?非引用類型有哪些

  • 非引用類型:保存在棧內(nèi)存中的簡(jiǎn)單數(shù)據(jù)段,數(shù)值浮驳,布爾值悍汛,null和undefined。

  • 引用類型: 指的是那些保存在堆內(nèi)存中的對(duì)象至会,變量中保存的實(shí)際上只是一個(gè)指針离咐,這個(gè)指針指向內(nèi)存中的另一個(gè)位置,由該位置保存對(duì)象,引用類型有對(duì)象宵蛀,數(shù)組昆著,函數(shù),正則术陶。

2.如下代碼輸出什么凑懂?為什么

var obj1 = {a:1, b:2};
var obj2 = {a:1, b:2};
console.log(obj1 == obj2);
console.log(obj1 = obj2);
console.log(obj1 == obj2);

輸出結(jié)果為

var obj1 = {a:1, b:2};
var obj2 = {a:1, b:2};
console.log(obj1 == obj2);  //false
console.log(obj1 = obj2); //{a:1,b:2}
console.log(obj1 == obj2); // true

解釋:第一行,obj1和obj2內(nèi)部保存的其實(shí)是兩個(gè)對(duì)象的地址梧宫,兩個(gè)對(duì)象是相互獨(dú)立的接谨,即不是同一個(gè)對(duì)象,即地址不相等塘匣,所以obj1 == obj2的值為false脓豪。
第二行和第三行,將obj2指向的對(duì)象的地址賦值給obj1馆铁,此時(shí)跑揉,obj1和obj2指向的是同一個(gè)對(duì)象,第二行輸出obj2指向的對(duì)象埠巨,第三行,obj1和obj2指向的已經(jīng)是同一個(gè)對(duì)象了现拒,所以輸出true辣垒。

3.如下代碼輸出什么? 為什么

 var a = 1
 var b = 2
 var c = { name: 'deejay', age: 2 }
 var d = [a, b, c]

var aa = a
var bb = b
var cc = c
var dd = d

a = 11
b = 22
c.name = 'hello'
d[2]['age'] = 3

console.log(aa) 
console.log(bb) 
console.log(cc)
console.log(dd)

輸出結(jié)果為:

    var a = 1
    var b = 2
    var c = { name: 'deejay', age: 2 }
    var d = [a, b, c]

    var aa = a
    var bb = b
    var cc = c
    var dd = d

    a = 11
    b = 22
    c.name = 'hello'
    d[2]['age'] = 3

    console.log(aa) //1
    console.log(bb) //2
    console.log(cc) //name:hello age:3
    console.log(dd) //1,2,name:hello  age: 3

基本類型和引用類型的不同,基本類型復(fù)制之后就是相對(duì)獨(dú)立的印蔬,引用類型僅僅傳遞了一個(gè)指向?qū)ο蟮牡刂费埃瑢?shí)質(zhì)上引用的還是同一個(gè)對(duì)象,所以通過(guò)指向這個(gè)對(duì)象的任何一個(gè)指針改變這個(gè)對(duì)象的屬性侥猬,都會(huì)引起所有變量的變化例驹。

4.如下代碼輸出什么? 為什么

var a = 1
var c = { name: 'deejay', age: 2 }

function f1(n){
  ++n
}
function f2(obj){
  ++obj.age
}

f1(a) 
f2(c) 
f1(c.age) 
console.log(a) 
console.log(c)

上述代碼可以理解為:

    var a = 1
    var c = { name: 'deejay', age: 2 }

    function f1(n){
//        var n = a;
//        var n = c.age
        ++n
    }
    function f2(obj){
//        var obj = c; 傳遞的是地址
        ++obj.age
    }

    f1(a)
    f2(c)
    f1(c.age)
    console.log(a)  //1
    console.log(c) //{ name: 'deejay', age: 3 }

調(diào)用f2的時(shí)候改變了c中的age

5.過(guò)濾如下數(shù)組,只保留正數(shù)退唠,直接在原數(shù)組上操作

var arr = [3,1,0,-1,-3,2,-5]
function filter(arr){
}
filter(arr)
console.log(arr) // [3,1,2]
    var arr = [3,1,0,-1,-3,2,-5]
    function filter(arr){
        for (var i = 0; i < arr.length; i ++) {
            if (arr[i] <= 0) {
                arr.splice(i,1);
                i -= 1; //因?yàn)閯h除了一位鹃锈,i的取值要減1
                // 也可以使用遞歸,直接調(diào)用 filter(arr);
            }
        }
        return arr;
    }
    filter(arr)
    console.log(arr) // [3,1,2]

6.過(guò)濾如下數(shù)組瞧预,只保留正數(shù)屎债,原數(shù)組不變,生成新數(shù)組

var arr = [3,1,0,-1,-3,2,-5]
function filter(arr){
}
var arr2 = filter(arr)
console.log(arr2) // [3,1,2]
console.log(arr)  // [3,1,0,-1,-2,2,-5]
    var arr = [3,1,0,-1,-3,2,-5]
    function filter(arr){
        var arr2 = [];  //創(chuàng)建新數(shù)組垢油,不會(huì)影響原數(shù)組
        for (var i =0; i < arr.length; i ++) {
            if (arr[i] > 0) {
                arr2.push(arr[i]);
            }
        }
        return  arr2;
    }
    var arr2 = filter(arr)
    console.log(arr2) // [3,1,2]
    console.log(arr)  // [3,1,0,-1,-2,2,-5]

7.寫一個(gè)深拷貝函數(shù)盆驹,用兩種方式實(shí)現(xiàn)

  • 遞歸方式:
    var oldObj = {
        name: 'deejay',
        age: 20,
        friend: {
            name: 'dee',
            age: 20,
        }
    }
    function deepCopy (oldObj) {
        var newObj = {};
        for (var key in oldObj) {
            if (oldObj.hasOwnProperty(key)) {
                if (typeof oldObj[key] === 'number' || typeof oldObj[key] === 'string' || typeof oldObj[key] ==='boolean' ||  oldObj[key] === null|| oldObj[[key]=== undefined]) {
                    newObj[key] = oldObj[key];
                } 
                else {
                    newObj[key] = deepCopy(oldObj[key]);
                }
            }
        }
        return newObj;
    }
    var newObj = deepCopy(oldObj);
    console.log(newObj);
  • JOSN方法實(shí)現(xiàn):
    var oldObj = {
        name: 'deejay',
        age: 20,
        friend: {
            name: 'dee',
            age: 20,
        }
    }
    function deepCopy (oldObj) {
        var newObj = {};
        newObj = JSON.parse(JSON.stringify(oldObj));
        return  newObj;
    }
    var newObj = deepCopy(oldObj);
    console.log(newObj);
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市滩愁,隨后出現(xiàn)的幾起案子躯喇,更是在濱河造成了極大的恐慌,老刑警劉巖硝枉,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件廉丽,死亡現(xiàn)場(chǎng)離奇詭異倦微,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)雅倒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門璃诀,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人蔑匣,你說(shuō)我怎么就攤上這事劣欢。” “怎么了裁良?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵凿将,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我价脾,道長(zhǎng)牧抵,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任侨把,我火速辦了婚禮犀变,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘秋柄。我一直安慰自己获枝,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布骇笔。 她就那樣靜靜地躺著省店,像睡著了一般。 火紅的嫁衣襯著肌膚如雪笨触。 梳的紋絲不亂的頭發(fā)上懦傍,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音芦劣,去河邊找鬼粗俱。 笑死,一個(gè)胖子當(dāng)著我的面吹牛持寄,可吹牛的內(nèi)容都是我干的源梭。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼稍味,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼废麻!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起模庐,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤烛愧,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體怜姿,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡慎冤,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了沧卢。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蚁堤。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖但狭,靈堂內(nèi)的尸體忽然破棺而出披诗,到底是詐尸還是另有隱情,我是刑警寧澤立磁,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布呈队,位于F島的核電站,受9級(jí)特大地震影響唱歧,放射性物質(zhì)發(fā)生泄漏宪摧。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一颅崩、第九天 我趴在偏房一處隱蔽的房頂上張望几于。 院中可真熱鬧,春花似錦沿后、人聲如沸孩革。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至锅移,卻和暖如春熔掺,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背非剃。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工置逻, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人备绽。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓券坞,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親肺素。 傳聞我的和親對(duì)象是個(gè)殘疾皇子恨锚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 1.引用類型有哪些?非引用類型有哪些 引用類型是指那些保存在堆內(nèi)存中的對(duì)象倍靡。變量中保存的實(shí)際上只是一個(gè)指針猴伶,這個(gè)指...
    高進(jìn)哥哥閱讀 315評(píng)論 0 0
  • 1.引用類型; 2.過(guò)濾數(shù)組; 3.深淺拷貝方法; 1.引用類型有哪些?非引用類型有哪些 引用類型:引用類型(Ob...
    好奇而已閱讀 1,364評(píng)論 0 0
  • 1. 引用類型有哪些?非引用類型有哪些 基本類型值(數(shù)值他挎、布爾值筝尾、null和undefined):指的是保存在棧內(nèi)...
    蕭雪圣閱讀 466評(píng)論 0 1
  • 1.引用類型有哪些?非引用類型有哪些2.如下代碼輸出什么办桨?為什么 引用類型有:對(duì)象, 數(shù)組, 函數(shù), 正則引用類型...
    billa_8f6b閱讀 299評(píng)論 0 0
  • 燦爛的五月[微笑]呢撞、火紅的五月[愉快]损姜、溫暖的五月[微笑],感謝在職黨員王老師精彩授課[擁抱]狸相,感謝一如既往支持我...
    廣仁閱讀 424評(píng)論 0 0