淺拷貝和深拷貝及區(qū)別

深拷貝與淺拷貝是針對(duì)對(duì)象屬性為對(duì)象的阳堕,因?yàn)榛緮?shù)據(jù)類型在進(jìn)行賦值操作時(shí)(也就是深拷貝(值拷貝))哩俭,是直接將值賦給了新的變量桦踊,也就是該變量是原變量的一個(gè)副本绞蹦,這時(shí)力奋,你修改兩個(gè)中的任意一個(gè)都不會(huì)影響另一個(gè);而對(duì)于對(duì)象或引用數(shù)據(jù)在進(jìn)行淺拷貝時(shí)坦辟,只是將對(duì)象的引用復(fù)制了一份刊侯,也就是內(nèi)存地址,即兩個(gè)不同的變量指向了同一個(gè)內(nèi)存地址锉走,那么在改變?nèi)我庖粋€(gè)變量的值都是改變內(nèi)存地址所存儲(chǔ)的值滨彻,因此兩個(gè)變量的值都會(huì)改變。

  1. js中的深拷貝(值拷貝):
    js 中的基本數(shù)據(jù)類型:String挪蹭,Number亭饵,Boolean,Null梁厉,Undefined辜羊,在賦值的過(guò)程中都是值拷貝。
    例如词顾,let a = 10; b = a 八秃,修改其中一個(gè)變量的值,不會(huì)影響到另一個(gè)變量的值肉盹。

  2. js 中的淺拷貝(引用拷貝)
    js 中的對(duì)象數(shù)據(jù)類型:Object昔驱,Array,F(xiàn)unction上忍,Map骤肛,Set,在賦值過(guò)程中都是引用拷貝窍蓝。

let obj = { 
    name:'小明',
    age: 18
}
let obj2 = obj
obj2.name = '小紅';
console.log(obj) // {name:'小紅',age: 18}
console.log(obj2) // {name:'小紅',age: 18}

當(dāng)修改obj2的name屬性時(shí)腋颠,也會(huì)修改obj的name,因?yàn)樗麄冎赶蛲粔K內(nèi)存地址吓笙。

  1. 將淺拷貝轉(zhuǎn)換為深拷貝
    在實(shí)際的項(xiàng)目開(kāi)發(fā)中淑玫,我們?cè)诙鄶?shù)情況下不希望將對(duì)象進(jìn)行淺拷貝,因?yàn)橹禃?huì)相互影響,容易出錯(cuò)混移。

3.1 Array的深拷貝

  1. 通過(guò)slice方法
    slice() 操作數(shù)組時(shí)祠墅,不會(huì)對(duì)原數(shù)組有影響,會(huì)產(chǎn)出一個(gè)新的數(shù)組歌径。
let arr1 = [1,72,5,6]
let arr2 = arr1.slice()
arr2[0] = 100
console.log(arr1)  // [1,72,5,6]
console.log(arr2)  // [100,72,5,6] 

數(shù)組arr2的改變未引起arr1的變化毁嗦。

  1. 通過(guò)concat方法
    數(shù)組的concat() 方法,能夠連接兩個(gè)數(shù)組回铛,同樣不會(huì)改變?cè)瓉?lái)的數(shù)組狗准。
    用一個(gè)空數(shù)組[ ] 連接另一個(gè)數(shù)組,即可實(shí)現(xiàn)深拷貝茵肃。
let arr3 = ['cat','dog','pig']
let arr4 = [].concat(arr3)
arr3[2] = 'big pig'
console.log(arr3)   // ['cat','dog','big pig']
console.log(arr4)   // ['cat','dog','pig']
  1. 通過(guò)ES6語(yǔ)法中的 …
    ES6中的 … 腔长,我們經(jīng)常在數(shù)組的深拷貝中用到。
let arr5 = [0,0,1]
let arr6 = [...arr5]
arr5[0] = 10000
console.log(arr5) // [10000,0,1]
console.log(arr6) // [0,0,1]
  1. 通過(guò)Array.from 方法
    Array.from() 方法能從一個(gè)類似數(shù)組或可迭代對(duì)象中返回一個(gè)新的數(shù)組實(shí)例验残。通過(guò)Array.from() 方法能獲取到一個(gè)數(shù)組的深拷貝捞附。
let arr7 = [1,2,3]
let arr8 = Array.from(arr7)
arr7[1] = 1000
console.log(arr7) // [1,1000,3]
console.log(arr8) // [1,2,3]

3.2 Object 的深拷貝

  1. 通過(guò)Object.assign() 方法
    ES6的Object.assign(target,…sources) 用于對(duì)象的合并,將源對(duì)象的所有可枚舉屬性您没,復(fù)制到目標(biāo)對(duì)象中鸟召,并返回合并后的目標(biāo)對(duì)象。后來(lái)的源對(duì)象的屬性值氨鹏,將會(huì)覆蓋它之前的對(duì)象的屬性欧募。
let person = {
    name: 'xia',
    age: 25,
    height: 160 
}
let otherPerson = Object.assign({},person)
person.age = 30
console.log(person) // {name: "xia", age: 30, height: 160}
console.log(otherPerson) // {name: "xia", age: 25, height: 160}
  1. 萬(wàn)能轉(zhuǎn)換器(對(duì)Array和Object)
    前面講了Array和Object的深拷貝方法,但是對(duì)于有更深層次的結(jié)構(gòu)關(guān)系(數(shù)組套數(shù)組仆抵,數(shù)組套對(duì)象跟继,對(duì)象套對(duì)象等),上面的方法就失靈了镣丑,可以看下面的例子舔糖。
let personArr = [{name: 'xia'}, {name: 'zhang'}]
let otherPersonArr2 = [...personArr]
personArr[0].name = 'xia xia'
console.log(personArr)
console.log(otherPersonArr2)

萬(wàn)能轉(zhuǎn)換器JSON.parse(JSON.stringify(obj)) 深拷貝已對(duì)象,它可以深拷貝多層級(jí)的莺匠,不同擔(dān)心嵌套問(wèn)題剩盒。

JSON.stringify() 將對(duì)象序列化成json對(duì)象
JSON.parse() 反序列化——將json對(duì)象反序列化成js對(duì)象
JSON.stringify(obj) 將js中的對(duì)象轉(zhuǎn)化為json字符串

let jack = {
    name: 'jack'
}
console.log(jack)
console.log(JSON.stringify(jack))

它們?cè)诟袷缴嫌袇^(qū)別。下圖中的第一個(gè)是對(duì)象慨蛙,name沒(méi)有雙引號(hào)括起來(lái)。第二個(gè)是json字符串纪挎,其中期贫,name用雙引號(hào)括起來(lái)了。

JSON.parse() 將json字符串解析成對(duì)象

 let obj = {
     name: '小明'
 }
 console.log('obj: ', obj)
 console.log('json string: ', JSON.stringify(obj))

 let str = JSON.stringify(obj)
 console.log('--------------')
 console.log(str)
 console.log('str to obj: ', JSON.parse(str))
  1. Vue中的淺拷貝和深拷貝
    兩個(gè)button-counter 共用同一個(gè)jack對(duì)象异袄,用同一塊地址通砍,當(dāng)其中一個(gè)實(shí)例改變時(shí),會(huì)影響另一個(gè)實(shí)例的值。(淺拷貝)
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>vue的data選項(xiàng)</title>
    <script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
    <div id="app">
        <button-counter></button-counter>
        <button-counter></button-counter>
    </div>

    <script>
        let jack = {
            counter: 0
        }
        // 子組件
        Vue.component('button-counter', {
            data() {
                // 函數(shù)類型
                return jack
            },
            template: `<button @click="counter++">click {{counter}} times</button>`
        })
        let vm = new Vue({
            el: '#app' // mount到DOM上
        })
    </script>
</body>
</html>

采用深拷貝封孙,重新創(chuàng)建一塊內(nèi)存迹冤。這樣,vue的button-counter組件中的counter值互不影響虎忌。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>vue的data選項(xiàng)</title>
    <script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
</head>
<body>
    <div id="app">
        <button-counter></button-counter>
        <button-counter></button-counter>
    </div>

    <script>
        let jack = {
            counter: 0
        }
        // 子組件
        Vue.component('button-counter', {
            data() {
                // 函數(shù)類型
                return JSON.parse(JSON.stringify(jack))
            },
            template: `<button @click="counter++">click {{counter}} times</button>`
        })
        let vm = new Vue({
            el: '#app' // mount到DOM上
        })
    </script>
</body>
</html>
``
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末泡徙,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子膜蠢,更是在濱河造成了極大的恐慌堪藐,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,914評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件挑围,死亡現(xiàn)場(chǎng)離奇詭異礁竞,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)杉辙,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評(píng)論 2 383
  • 文/潘曉璐 我一進(jìn)店門模捂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人蜘矢,你說(shuō)我怎么就攤上這事狂男。” “怎么了硼端?”我有些...
    開(kāi)封第一講書人閱讀 156,531評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵并淋,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我珍昨,道長(zhǎng)县耽,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 56,309評(píng)論 1 282
  • 正文 為了忘掉前任镣典,我火速辦了婚禮兔毙,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘兄春。我一直安慰自己澎剥,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,381評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布赶舆。 她就那樣靜靜地躺著哑姚,像睡著了一般。 火紅的嫁衣襯著肌膚如雪芜茵。 梳的紋絲不亂的頭發(fā)上叙量,一...
    開(kāi)封第一講書人閱讀 49,730評(píng)論 1 289
  • 那天,我揣著相機(jī)與錄音九串,去河邊找鬼绞佩。 笑死寺鸥,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的品山。 我是一名探鬼主播胆建,決...
    沈念sama閱讀 38,882評(píng)論 3 404
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼肘交!你這毒婦竟也來(lái)了笆载?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 37,643評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤酸些,失蹤者是張志新(化名)和其女友劉穎宰译,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體魄懂,經(jīng)...
    沈念sama閱讀 44,095評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡沿侈,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,448評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了市栗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片缀拭。...
    茶點(diǎn)故事閱讀 38,566評(píng)論 1 339
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖填帽,靈堂內(nèi)的尸體忽然破棺而出蛛淋,到底是詐尸還是另有隱情,我是刑警寧澤篡腌,帶...
    沈念sama閱讀 34,253評(píng)論 4 328
  • 正文 年R本政府宣布褐荷,位于F島的核電站,受9級(jí)特大地震影響嘹悼,放射性物質(zhì)發(fā)生泄漏叛甫。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,829評(píng)論 3 312
  • 文/蒙蒙 一杨伙、第九天 我趴在偏房一處隱蔽的房頂上張望其监。 院中可真熱鬧,春花似錦限匣、人聲如沸抖苦。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,715評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)锌历。三九已至,卻和暖如春峦筒,著一層夾襖步出監(jiān)牢的瞬間辩涝,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,945評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工勘天, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,248評(píng)論 2 360
  • 正文 我出身青樓脯丝,卻偏偏與公主長(zhǎng)得像商膊,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子宠进,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,440評(píng)論 2 348