亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

詳細分析JavaScript中的深淺拷貝

 更新時間:2020年09月17日 10:50:41   作者:TanJia  
這篇文章主要介紹了JavaScript中的深淺拷貝,幫助大家更好的理解和學(xué)習(xí)JavaScript,感興趣的朋友可以了解下

在說JS中深淺拷貝之前,我們需要對JS中的數(shù)據(jù)類型有所了解,分為基本數(shù)據(jù)類型與引用數(shù)據(jù)類型,對于基本數(shù)據(jù)類型并沒有深淺拷貝的說法,深淺拷貝主要針對引用數(shù)據(jù)類型。

一、淺拷貝

淺拷貝只復(fù)制了引用,并沒有復(fù)制值。在JS中最簡單的淺拷貝就是利用“=”賦值操作符來實現(xiàn)。

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'},
  fun:function(){
    console.log('fun')
  }
}
var obj2 = obj1
obj2.a = 666 /* 修改obj2的值,obj1的值也隨之改變 */
console.log(obj1) /* {a: 666, b: Array(3), c: {…}, fun: ƒ} */

上述代碼中,我們修改obj2的值,obj1的值也隨之發(fā)生了改變,使用”=“只實現(xiàn)了淺拷貝。

二、深拷貝

深拷貝是對目標的完全拷貝,進行深拷貝后的兩個值互不影響。

1. 利用JSON.stringify與JSON.parse方法

JSON.stringify將一個JavaScript值轉(zhuǎn)換為JSON字符串;

JSON.parse將一個JSON字符串轉(zhuǎn)換為JavaScript值。

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'},
}			
var obj2 = JSON.parse(JSON.stringify(obj1))
obj2.a = 12
console.log(obj1) /* {a: 1, b: Array(3), c: {…}} */

修改obj2的值并沒有影響到obj1中的屬性值,顯然,我們利用JSON.parse與JSON.stringify實現(xiàn)了深拷貝。

但是,真的可以這么簡單的實現(xiàn)嗎?我們來看看下面的例子!

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'},
  fun:function(){
    console.log('fun')
  }
}			
var obj2 = JSON.parse(JSON.stringify(obj1))
obj2.a = 12
console.log(obj1) /* {a: 1, b: Array(3), c: {…}, fun: ƒ} */
console.log(obj2) /* {a: 12, b: Array(3), c: {…}} */

轉(zhuǎn)換后的obj2中沒有了fun這個屬性,這是由于在利用JSON.stringify轉(zhuǎn)換過程中,忽略了undefined、function、symbol。顯然,當(dāng)我們的對象中出現(xiàn)這些類型的屬性時無法利用該方法實現(xiàn)深拷貝。

2. 遞歸

function deepClone(source){
  if(!isObject(source)) return source
  var newObj = source instanceof Array? []:{}
  for(let key in source){
	if(source.hasOwnProperty(key)){
	  newObj[key] = isObject(source[key])?deepClone(source[key]):source[key]
    }
  }
  return newObj
}
function isObject(x) {
  return typeof x === 'object' && x != null
}

測試一下上述方法:

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'},
  fun:function(){
	console.log('fun')
  }
}			
var obj2 = deepClone(obj1)
obj2.a = 12
console.log(obj1) /* {a: 1, b: Array(3), c: {…}, fun: ƒ} */

通過例子可以看到,我們修改了obj2中a屬性的值,但是并沒有影響到obj1中的a屬性值。通過遞歸我們可以實現(xiàn)深拷貝!

注意:上述方法未解決循環(huán)引用的問題。

var obj1 = {}
obj1.a = obj1
var obj2 = deepClone(obj1) /* 報錯,棧溢出 */
console.log(obj2)

關(guān)于如何解決循環(huán)引用問題以及實現(xiàn)Symbol類型拷貝,稍后完善。

三、其他拷貝方法

1. 數(shù)組中的concat()和slice()方法

我們知道數(shù)組中有兩個方法concat和slice可以完成復(fù)制數(shù)組,并且返回新數(shù)組。以concat為例。

var arr = [1,2,3]
var arr2 = arr.concat()
arr2[2]=4
console.log(arr) /* [1, 2, 3] */
console.log(arr2) /* [1, 2, 4] */

改變arr2的值,并沒有影響到arr的值,這是實現(xiàn)了數(shù)組的深拷貝嗎,先不急于下結(jié)論,一起看看下面的例子再來分析:

var arr = [1,2,3,[4,5,6],{a:7}]
var arr2 = arr.concat()
arr2[3] = 444
arr2[4].a=8
console.log(arr) /* [1,2,3,[4,5,6],{a:8}] */
console.log(arr2) /* [1,2,3,444,{a:8}] */

我們直接修改arr2[3],并沒有引起arr的改變,但是我們修改arr2[4].a時,arr中的相應(yīng)元素跟著一起發(fā)生了改變。其實,我們對arr2數(shù)組中的元素直接進行改變(比如:arr2[0]=***,arr2[1]=***,arr2[3]=***)時,不會影響到原數(shù)組arr,但是我們修改數(shù)組中的[3,4,5]或{a:7}時,會造成原數(shù)組arr的改變。

結(jié)論:concat()方法對數(shù)組第一層進行了深拷貝。

可以再試試數(shù)組的slice()方法,它也是只對數(shù)組第一層進行了深拷貝。

2. Object.assign()和...展開運算符

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'}
}
var obj2 = {...obj1}
obj2.a = 666
obj2.c.name = 'xinxin'
console.log(obj1) /* {a:1,b:[2,3,4],c:{name:'xinxin'}} */

可以看到利用...展開運算符實現(xiàn)的是對象第一層的深拷貝。后面的只是拷貝的引用值。

可以試試Object.assign()方法:

var obj1 = {
  a:1,
  b:[2,3,4],
  c:{name:'tanj'}
}
var obj2 = {}
Object.assign(obj2,obj1)
obj2.a = 666
obj2.b[0] = 0
console.log(obj1) /* {a:1,b:[0,3,4],c:{name:'tanj'} */

同樣,只對對象第一層進行了深拷貝,假如源對象的屬性值(例如obj1)是一個指向?qū)ο蟮囊?,obj2也只拷貝那個引用值。所以改變obj2中b所指向的那個數(shù)組時,obj1的值也會發(fā)生改變。

我們可以自己實現(xiàn)一個這樣的效果,只對第一層進行深拷貝:

function shallowClone(source) {
  const newObj = source.constructor === Array ? [] : {}
  for (let keys in source) {
    if (source.hasOwnProperty(keys)) {
	  newObj[keys] = source[keys]
	}
  }
  return newObj
}

以上就是分析JavaScript中的深淺拷貝的詳細內(nèi)容,更多關(guān)于JavaScript 深淺拷貝的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • javascript實現(xiàn)復(fù)選框全選或反選

    javascript實現(xiàn)復(fù)選框全選或反選

    這篇文章主要為大家詳細介紹了javascript實現(xiàn)復(fù)選框全選或反選,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-02-02
  • 使用 Javascript 實現(xiàn)瀏覽器推送提醒功能的示例

    使用 Javascript 實現(xiàn)瀏覽器推送提醒功能的示例

    本篇文章主要介紹了使用 Javascript 實現(xiàn)瀏覽器推送提醒功能的示例,具有一定的參考價值,有興趣的可以了解一下
    2017-11-11
  • TypeScript接口和類型的區(qū)別小結(jié)

    TypeScript接口和類型的區(qū)別小結(jié)

    在 TypeScript 中,有兩種主要的定義自定義類型的方式,接口和類型,盡管它們在外觀上可能相似,但它們之間有一些關(guān)鍵的區(qū)別,本文就詳細的介紹一下,感興趣的可以了解下
    2023-05-05
  • js活用事件觸發(fā)對象動作

    js活用事件觸發(fā)對象動作

    現(xiàn)在基本上我很少在對象里面寫事情觸發(fā)行為了,因為感覺那樣做相同的對象代碼利用率太低,容易導(dǎo)致代碼冗余.
    2008-08-08
  • 微信小程序input框中加入小圖標的實現(xiàn)方法

    微信小程序input框中加入小圖標的實現(xiàn)方法

    這篇文章主要介紹了微信小程序input框中加入小圖標的實現(xiàn)方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2018-06-06
  • 從零開始做一個pagination分頁組件

    從零開始做一個pagination分頁組件

    從零開始做一個pagination分頁組件,這篇文章主要介紹了pagination分頁組件的制作方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-03-03
  • 在JavaScript中實現(xiàn)類的方式探討

    在JavaScript中實現(xiàn)類的方式探討

    在 javascript 中有很多方式來創(chuàng)建對象,所以創(chuàng)建對象的方式使用起來非常靈活,到底哪一種方式是最恰當(dāng)呢?下面為大家講講
    2013-08-08
  • 輕松理解JavaScript閉包

    輕松理解JavaScript閉包

    閉包機制是JavaScript的重點和難點,本文希望能幫助大家輕松的學(xué)習(xí)閉包。下面跟著小編一起來看下吧
    2017-03-03
  • 利用JS輕松實現(xiàn)獲取表單數(shù)據(jù)

    利用JS輕松實現(xiàn)獲取表單數(shù)據(jù)

    本文主要介紹了利用JS輕松實現(xiàn)獲取表單數(shù)據(jù)。方法有別于原始的做法,大家可以試著找原始做法與本文所說方法之間的區(qū)別。有興趣的朋友可以看下,希望對大家有所幫助
    2016-12-12
  • js神秘的電報密碼 哈弗曼編碼實現(xiàn)

    js神秘的電報密碼 哈弗曼編碼實現(xiàn)

    這篇文章主要介紹了js神秘的電報密碼 哈弗曼編碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-09-09

最新評論