精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

精學手撕系列——深淺拷貝原理

開發
本文是作者在準備面試時,精心準備的深淺拷貝—精講系列文章,希望可以幫助到大家

 一.JS中淺拷貝的手段有哪些?

[[342550]]


1.什么是拷貝?
我們來看下面一個例子,幫助大家區分賦值與拷貝的區別:

  1. let arr = [1, 2, 3]; 
  2. let newArr = arr; 
  3. newArr[0] = 100; 
  4.  
  5. console.log(arr); // [100, 2, 3] 

這是直接賦值的情況,不涉及任何拷貝。當改變newArr的時候,由于是同一個引用,arr指向的值也跟著改變。

現在進行淺拷貝:

  1. let arr = [1, 2, 3]; 
  2. let newArr = arr.slice(); 
  3. newArr[0] = 100; 
  4.  
  5. console.log(arr); //[1, 2, 3] 

當我們修改newArr的時候,arr的值并不改變,這是因為newArr是arr淺拷貝后的結果,newArr和arr現在是兩個不同的引用地址了。

但我們再來看一個潛在的問題:

  1. let arr = [1, 2, {val: 4}]; 
  2. let newArr = arr.slice(); 
  3. newArr[2].val = 1000; 
  4.  
  5. console.log(arr);//[ 1, 2, { val: 1000 } ] 

這里我們明顯看到,淺拷貝雖然只能復制一層內容。但如果復制的第一層內容中,有復雜數據類型(數組/對象),那么淺拷貝將失效,這也是淺拷貝最大的限制所在了。

但幸運的是,深拷貝就是為了解決這個問題而生的,它能解決無限層級的對象嵌套問題,實現徹底的拷貝。深拷貝我們在下一道題中介紹。

接下來,我們來歸納一下JS中實現的淺拷貝都有哪些方法呢?

2.手動實現

  1. const shallClone = (target) => { 
  2.   if (typeof target === 'object' && target !== null) { 
  3.     const cloneTarget = Array.isArray(target) ? [] : {}; 
  4.     for (let prop in target) { 
  5.       if (target.hasOwnProperty(prop)) { // 遍歷對象自身可枚舉屬性(不考慮繼承屬性和原型對象) 
  6.         cloneTarget[prop] = target[prop]; 
  7.     } 
  8.     return cloneTarget; 
  9.   } else { 
  10.     return target; 
  11.   } 

3.Object.assign
但是需要注意的是,Object.assgin() 拷貝的是對象的屬性的引用,而不是對象本身。

  1. let obj = { name'sy', age: 18 }; 
  2. const obj2 = Object.assign({}, obj, {Newname: 'sss'}); 
  3. console.log(obj2);  // {name"sy", age: 18, Newname: "sss"

4.concat()淺拷貝數組

  1. let arr = [1, 2, 3]; 
  2. let newArr = arr.concat(); 
  3. newArr[1] = 100; 
  4. console.log(arr); //[ 1, 2, 3 ] 

5.slice()淺拷貝
開頭例子就是!!

6. ...展開運算符

  1. let arr = [1, 2, 3]; 
  2. let newArr = [...arr]; //跟arr.slice()是一樣的效果 

二.JS中深拷貝的手段有哪些?
在實現一個完整版的深拷貝函數之前,看看有沒有某個api能幫助我們完成深拷貝?

1.api版-簡易版

  1. JSON.parse(JSON.stringify()); 

從下面例子中,我們可以看出簡易版的深拷貝,已經做到了。

  1. let arr = [10, [100, 200], { x: 10, y:20}]; 
  2. let newArr = JSON.parse(JSON.stringify(arr));  
  3. console.log(newArr[2] === arr[2]; //  false 

其實,上面的api,它所使用的是暴力法,什么是暴力法呢,在這里給大家解釋一下:

暴力法:把原始數據直接變為字符串,再把字符串變為對象,(此時瀏覽器會重新開辟所有的內存空間),實現深拷貝。

但是直接供我們使用的api,往往會有一些自己的弊端,比如我們看下面這個例子

  1. let obj = { 
  2.   a: 100, 
  3.   b: [10, 20, 30], 
  4.   c: { 
  5.     x: 10 
  6.   }, 
  7.   d: /^\d+$/, 
  8.    // d: function() {} 
  9.   // d: new Date() 
  10.   // d: BigInt('10'
  11.   // d: Symbol('f'
  12. }; 
  13.  
  14. let newObj = JSON.parse(JSON.stringify(obj)); 
  15. console.log(newObj); 
  16. /* 
  17.   {a: 100, b: Array(3), c: {…}, d: {…}} 
  18.     a: 100 
  19.     b: (3) [10, 20, 30] 
  20.     c: {x: 10} 
  21.     d: {} 
  22.     __proto__: Object 
  23.   } 
  24. */ 

從上面例子的輸出結果中,我們可以看出,正則屬性直接變成了空對象。

那假如我們再把最后一個屬性,換成其它類型試一試,我們同理也可以發現,JSON.parse(JSON.stringify()) 都是有弊端的。我們總結一下:

正則屬性會變為空對象
函數會直接消失
日期直接字符串
Symbol直接消失
BigInt('10'),直接報錯
undefined會直接消失
所以當對象中沒有以上形式的屬性時,可以用JSON.parse(JSON.stringify())。

但是此方法還有一個弊端,那就是循環引用問題,舉個例子:

  1. const a = {value: 2}, 
  2. a.target = a; 

拷貝a會出現系統棧溢出,因為出現了無限遞歸的情況。

2.實現簡易版深拷貝
發現上面弊端后,我們來手寫一版深拷貝

此種方法,不考慮循環引用問題,也不考慮特殊對象的問題

  1. function deepClone(target)  { 
  2.   if (target === nullreturn null
  3.   if (typeof target !== 'object'return target; 
  4.  
  5.   const cloneTarget = Array.isArray(target) ? [] : {}; 
  6.   for (let prop in target) { 
  7.     if (target.hasOwnProperty(prop)) { 
  8.       cloneTarget[prop] = deepClone(target[prop]); 
  9.     } 
  10.   } 
  11.   return cloneTarget; 

現在我們以發現的幾個問題為導向,依次完善深拷貝函數

3.解決循環引用
問題如下:

  1. let obj = { value: 100 }; 
  2. obj.target = obj; 
  3.  
  4. deepClone(obj); //報錯: RangeError: Maximum call stack size exceeded 

這就是循環引用。我們怎么來解決這個問題呢?

創建一個Map(Map類似于對象,也是鍵值對的集合,但是“鍵”可以是對象),記錄下已經拷貝過的對象,如果已經拷貝過,那直接返回就行了.

其原理是每次拷貝引用類型的時候,都設置一個true作為標記,等下次再遍歷該對象的時候,就知道它是否已經拷貝過。

  1. const isObject = (target) => (typeof target === 'object' || typeof target === 'function') && target !== null
  2.  
  3. function deepClone (target, map = new Map()) { 
  4.   // 先判斷該引用類型是否被 拷貝過 
  5.   if (map.get(target)) { 
  6.     return target; 
  7.   } 
  8.  
  9.   if (isObject(target)) { 
  10.     map.set(target, true); 
  11.     const cloneTarget = Array.isArray(target) ? [] : {}; 
  12.     for (let prop in target) { 
  13.       if (target.hasOwnProperty(props)) { 
  14.         cloneTarget[prop] = deepClone(target[props], map); 
  15.       } 
  16.     } 
  17.     return cloneTarget; 
  18.   } else { 
  19.     return target; 
  20.   } 

現在我們就可以到已經成功了:

  1. const a = {val:2}; 
  2. a.target = a; 
  3. let newA = deepClone(a); 
  4. console.log(newA)//{ val: 2, target: { val: 2, target: [Circular] } } 

好像是沒有問題了, 拷貝也完成了。但還是有一個潛在的坑, 就是map 上的 key 和 map 構成了強引用關系,這是相當危險的。我給你解釋一下與之相對的弱引用的概念你就明白了:

在計算機程序設計中,弱引用與強引用相對, 是指不能確保其引用的對象不會被垃圾回收器回收的引用。一個對象若只被弱引用所引用,則被認為是不可訪問(或弱可訪問)的,并因此可能在任何時刻被回收。--百度百科

說的有一點繞,我用大白話解釋一下,被弱引用的對象可以在任何時候被回收,而對于強引用來說,只要這個強引用還在,那么對象無法被回收。拿上面的例子說,map 和 a一直是強引用的關系,在程序結束之前,a所占的內存空間一直不會被釋放,便會造成嚴重的內存泄漏問題。

怎么解決這個問題呢?

很簡單,讓 map 的 key 和 map 構成弱引用即可。ES6給我們提供了這樣的數據結構,它的名字叫WeakMap,它是一種特殊的Map, 其中的鍵是弱引用的。其鍵必須是對象,而值可以是任意的。

稍微改造一下極課:

  1. const deepClone = (target, map = new WeakMap()) => { 
  2.   //... 

4.解決特殊對象問題(RegExp,Date...)
如果傳入的對象格式滿足,正則或日期格式的話,返回一個新的正則或日期對象的實例

  1. function deepClone (target, map = new Map()) { 
  2.   
  3.   // 檢測當前對象target是否與 正則、日期格式對象匹配 
  4.   if (/^(RegExp|Date)$/i.test(target.constructor.name)){ 
  5.     new constructor(target); 
  6.   } 

5.完整版深克隆實現源碼

  1. const isObject = (target) => (typeof target === 'object' || typeof target === 'function') && target !== null
  2.  
  3. function deepClone (target, map = new Map()) { 
  4.   // 先判斷該引用類型是否被 拷貝過 
  5.   if (map.get(target)) { 
  6.     return target; 
  7.   } 
  8.  
  9.   // 檢測當前對象target是否與 正則、日期格式對象匹配 
  10.   if (/^(RegExp|Date)$/i.test(target.constructor.name)){ 
  11.     new constructor(target); 
  12.   } 
  13.  
  14.   if (isObject(target)) { 
  15.     map.set(target, true); 
  16.     const cloneTarget = Array.isArray(target) ? [] : {}; 
  17.     for (let prop in target) { 
  18.       if (target.hasOwnProperty(props)) { 
  19.         cloneTarget[prop] = deepClone(target[props], map); 
  20.       } 
  21.     } 
  22.     return cloneTarget; 
  23.   } else { 
  24.     return target; 
  25.   } 

補充:Object.keys(obj)只遍歷私有屬性(原型上可能有公共的方法,無法遍歷)

 

責任編輯:姜華 來源: 前端時光屋
相關推薦

2020-09-16 14:17:42

flat方法

2021-07-15 14:29:06

LRU算法

2021-09-06 08:13:35

APM系統監控

2011-06-03 17:14:35

iphone Objective

2020-12-01 11:33:57

Python拷貝copy

2023-05-12 08:11:58

JavaScriptJSON克隆

2022-09-26 09:01:23

JavaScript淺拷貝深拷貝

2023-11-08 14:21:51

Python拷貝

2024-12-03 16:49:58

2020-09-15 08:55:07

算法數據基礎

2021-01-07 08:29:46

Java淺拷貝深拷貝

2021-10-31 07:38:37

排序算法代碼

2023-09-18 09:10:11

Golang高性能緩存庫

2020-09-25 16:40:52

Selenium

2021-06-09 07:01:30

前端CallApply

2020-12-18 06:09:07

Java淺拷貝深拷貝

2021-05-18 07:52:31

PromiseAsyncAwait

2015-11-06 11:02:24

微信羅素生活

2024-08-06 10:16:52

Java AgentJava

2020-09-27 08:04:57

前端
點贊
收藏

51CTO技術棧公眾號

午夜免费福利在线观看| 成人公开免费视频| 伊人久久亚洲| 精品久久久久久亚洲国产300 | 国产精品视频二| 青青青草网站免费视频在线观看| 奇米色一区二区| 欧美肥老妇视频| 亚洲黄色小说视频| 中文字幕一区日韩精品| 色婷婷国产精品久久包臀| 亚洲一区二区三区欧美| 日韩有码第一页| 久久99热国产| 日本精品一区二区三区在线播放视频| 老司机成人免费视频| 日韩成人av在线资源| 欧美丰满少妇xxxxx高潮对白| 免费不卡av在线| 日韩黄色影院| 久久综合九色综合欧美98| 91九色在线视频| 在线永久看片免费的视频| 国产一区观看| www.欧美精品| 手机免费看av| 九九热hot精品视频在线播放| 欧美精品在线观看播放| 99久久激情视频| 美女的胸无遮挡在线观看| 亚洲色大成网站www久久九九| 欧美专区一二三| 神宫寺奈绪一区二区三区| 国产乱对白刺激视频不卡| 国产精品久久久久久久电影| 西西44rtwww国产精品| 欧美人成在线| 久久躁日日躁aaaaxxxx| 九九九视频在线观看| 亚洲免费观看高清完整版在线观| 精品日韩欧美一区二区| 午夜大片在线观看| 亚洲精品乱码日韩| 欧美无砖砖区免费| 亚洲黄色小视频在线观看| 中文字幕在线直播| 狠狠综合久久av一区二区小说| 日本男女交配视频| 青草在线视频| 一个色综合网站| 大胆欧美熟妇xx| 欧美性受ⅹ╳╳╳黑人a性爽| 一区免费观看视频| 在线日韩av永久免费观看| 成人动漫在线免费观看| 国产片一区二区三区| 欧美日韩国产免费一区二区三区| 天天色天天操天天射| 99久久精品一区| 久久av一区二区三区漫画| 天天射,天天干| 99精品久久只有精品| 久久伦理网站| 成年人在线视频| 亚洲欧洲精品一区二区三区不卡| 激情图片qvod| 性欧美video高清bbw| 亚洲午夜在线视频| 黑森林福利视频导航| 日韩电影大全网站| 欧美日韩午夜影院| xxx中文字幕| 成人资源在线播放| 亚洲精品av在线| 娇妻被老王脔到高潮失禁视频| 成人3d动漫在线观看| 日韩中文综合网| 高h视频免费观看| 国产综合自拍| 热久久美女精品天天吊色| 国产一区免费看| 精东粉嫩av免费一区二区三区| 亚洲va欧美va国产综合久久| 六月婷婷综合网| 久久久久久久久99精品| 一区二区三区|亚洲午夜| 黄色羞羞视频在线观看| 色呦呦日韩精品| 天堂在线一区二区三区| 1313精品午夜理伦电影| 亚洲欧洲日本专区| 欧美日韩午夜视频| 91久久亚洲| 国产精品免费一区| 俄罗斯嫩小性bbwbbw| 久久久久国产一区二区三区四区 | 欧美mv日韩| 欧美激情在线一区| 人人草在线观看| 国产成人高清视频| 日韩欧美一区二区三区久久婷婷| 成年人网站在线| 欧美日韩中文在线观看| 99九九99九九九99九他书对| 九九热播视频在线精品6 | 中文字幕资源网在线观看| 欧美日韩一区二区在线 | 国产激情偷乱视频一区二区三区| 精品中文字幕一区| a级在线观看| 欧美天堂亚洲电影院在线播放| 第一页在线视频| 久久中文亚洲字幕| 日本久久久久久久| 亚洲美女综合网| 亚洲欧洲日韩在线| 欧美少妇性生活视频| 777久久精品| 久久影视电视剧免费网站| 国产精品久久久久久人| 成人一区二区三区视频| 中文字幕av日韩精品| 亚洲伦乱视频| 亚洲精品久久久久久久久久久 | 亚洲高清在线精品| 嫩草视频免费在线观看| 伊人久久大香线蕉无限次| 久久久噜噜噜久久久| 国产麻豆91视频| 国产精品女主播av| 免费看a级黄色片| 日韩深夜影院| 91国语精品自产拍在线观看性色| 国产黄色片av| 亚洲黄色小视频| 国产精品嫩草影院8vv8 | 国产成人激情av| 国产盗摄视频在线观看| 成人国产精品久久| 久久天天躁狠狠躁老女人| 亚洲一区二区色| 中文成人综合网| 鲁一鲁一鲁一鲁一av| 精品免费视频| 国产精品美乳一区二区免费| 搞黄视频免费在线观看| 在线观看成人免费视频| 国产三级av在线播放| 久久久一二三| 日韩福利影院| 色诱色偷偷久久综合| 在线播放精品一区二区三区| 中文在线最新版天堂| 国产色产综合产在线视频| 久久婷婷国产91天堂综合精品| 久久不见久久见免费视频7| 欧美在线激情网| 成人av电影观看| 91精品啪在线观看国产60岁| 欧美成人一区二区三区高清| 丁香六月综合激情| 91国视频在线| 国际精品欧美精品| 91精品在线播放| 欧美78videosex性欧美| 亚洲精品久久久久久久久久久| 69视频免费在线观看| 国产欧美视频一区二区| 天堂在线中文在线| 国产一区观看| 日本成人三级电影网站| 欧美一级做a| 欧美精品免费在线观看| 色呦呦中文字幕| 91成人免费在线视频| 国产精品夜夜夜爽阿娇| 粉嫩aⅴ一区二区三区四区| 国产亚洲欧美在线视频| 成人情趣视频| 亚洲综合色激情五月| 蜜桃麻豆影像在线观看| 亚洲人成网站免费播放| 国产又粗又猛又爽又黄91| 一区二区三区四区高清精品免费观看 | 日韩和欧美的一区| 精品久久免费观看| 美女视频免费精品| 国产精品天天狠天天看| 蜜臀av国内免费精品久久久夜夜| 亚洲免费av电影| 999久久久久| 色综合久久综合中文综合网| 久久99久久99精品免费看小说| 成人精品鲁一区一区二区| 国产wwwxx| 亚洲人www| 一区二区精品在线观看| 人人精品视频| 亚洲一区二区中文字幕| 吞精囗交69激情欧美| 欧美另类老女人| 成人免费在线电影| 亚洲电影av在线| 久久青青草原亚洲av无码麻豆 | 亚洲男人天堂手机在线| 国产毛片毛片毛片毛片| 色婷婷久久久久swag精品| 欧美色图亚洲天堂| 国产精品乱人伦中文| 性久久久久久久久久久| 激情综合网最新| 黄色一级免费大片| 一本久道久久综合婷婷鲸鱼| 日韩 欧美 自拍| 欧美色图激情小说| 精品网站在线看| 国产精一区二区| 国产精品久久久久久久久久尿| 91九色国产在线播放| 欧美成人合集magnet| 美女隐私在线观看| 亚洲桃花岛网站| 四虎在线观看| 亚洲精品mp4| 成人免费视频国产免费麻豆| 在线不卡免费欧美| 亚洲av无码乱码国产精品fc2| 午夜精品久久久久影视| 久久无码精品丰满人妻| 一区精品在线播放| 蜜桃视频最新网址| 国产精品免费免费| av网站免费在线看| 久久一夜天堂av一区二区三区| 三级视频网站在线观看| 丁香网亚洲国际| 免费黄色在线播放| 国产不卡高清在线观看视频| 国产成人在线综合| 精品一区二区三区视频在线观看 | 国产一区二区不卡老阿姨| 爱爱爱爱免费视频| 久久99久久久久| 九九九九九国产| 久久国产精品免费| av在线网址导航| 国内不卡的二区三区中文字幕| 九九热精品在线播放| 蜜臀精品久久久久久蜜臀| www.日日操| 蜜臀a∨国产成人精品| 杨幂毛片午夜性生毛片| 免播放器亚洲一区| 国产精品探花在线播放| 国产一区二区网址| 亚洲一二三四五| 不卡视频在线观看| 亚洲天堂久久新| 亚洲国产精品传媒在线观看| 四虎成人免费影院| 日韩美女精品在线| 久久黄色免费网站| 欧美日韩一区二区精品| 中文字幕免费观看| 欧美日韩一区二区在线观看视频 | 亚洲成人黄色网址| 亚洲人成色777777老人头| 亚洲欧美成人网| 亚洲成人三级| 欧美俄罗斯性视频| 亚洲精品日产| 国产中文字幕91| 久久影院一区二区三区| 国产一区二区三区高清| 羞羞色国产精品网站| 神马影院午夜我不卡影院| 亚洲精品一区二区在线看| 特色特色大片在线| 国产精品一二| 岛国av免费在线| 丁香啪啪综合成人亚洲小说| 免费看污黄网站在线观看| 中文字幕精品在线不卡| 波多野结衣不卡视频| 国产www免费| 欧美丝袜足交| 香蕉久久夜色| 欧美日韩亚洲一区三区| 男人天堂999| 麻豆精品在线观看| 岛国av免费观看| 亚洲国产精品成人久久综合一区| 手机在线免费看毛片| 精品国产成人在线| 一级黄色大片网站| 亚洲国产精品人久久电影| 成人影视在线播放| 欧美精品电影免费在线观看| 精品成人免费一区二区在线播放| 91在线精品播放| 久久超碰99| www.av91| 韩国欧美国产1区| 日韩中文字幕电影| 一区二区高清免费观看影视大全| 91在线视频免费播放| 日韩欧美一二三四区| 北岛玲一区二区三区| 久久久综合av| 中文幕av一区二区三区佐山爱| 久久av一区二区三区漫画| 中文字幕亚洲精品乱码| 天天色综合天天色| 久久新电视剧免费观看| 免费日韩在线视频| 欧美日本一道本| 黄色片在线看| 97精品在线视频| 一区二区精彩视频| www.午夜色| 日本网站在线观看一区二区三区 | 一二三区免费视频| 精品播放一区二区| www久久日com| 国产在线日韩在线| 日韩精品免费一区二区在线观看| 99热成人精品热久久66| heyzo一本久久综合| 欧美成欧美va| 欧美一级黄色片| 福利视频在线| 91久久国产精品| 小处雏高清一区二区三区| 黄色片视频在线| 亚洲国产精品二十页| 精品乱码一区内射人妻无码| 日韩精品视频在线播放| 成人影院在线视频| 精品国产一区二区三区四区vr| 精品白丝av| 怡红院一区二区| 亚洲电影一区二区三区| 亚洲欧美国产高清va在线播放| 久久亚洲综合国产精品99麻豆精品福利 | 亚洲乱码国产乱码精品精可以看| 中文字幕在线观看你懂的| 一区二区欧美亚洲| 91超碰碰碰碰久久久久久综合| 日本一区二区视频| 日韩成人伦理电影在线观看| 日本二区在线观看| 在线观看91精品国产入口| 成人网视频在线观看| 国产精品久久久久久久久久久新郎 | 欧美一区二区三区四区视频| 大地资源网3页在线观看| 91久久久一线二线三线品牌| 亚洲国产一成人久久精品| 国产精品一级无码| 亚洲sss视频在线视频| 天堂av在线7| 国产男女猛烈无遮挡91| 午夜精品视频一区二区三区在线看| 欧美成人手机在线视频| 亚洲一区二区精品视频| 日韩私人影院| 国产精品劲爆视频| 91成人国产| 白丝校花扒腿让我c| 欧美视频一二三| 一本一道波多野毛片中文在线| 亚洲在线视频福利| 一本色道久久综合亚洲精品不| 精品人妻互换一区二区三区| 欧美日韩一区小说| 1234区中文字幕在线观看| 欧美精品免费观看二区| 久久成人麻豆午夜电影| 欧美国产日韩综合| 精品丝袜一区二区三区| 欧美一级二级视频| www.欧美黄色| 国产亚洲短视频| va婷婷在线免费观看| 26uuu国产精品视频| 色小子综合网| 中文字幕第九页| 欧美中文字幕一区| 色综合999| 日韩午夜视频在线观看| 国产精品综合在线视频| 国产成人无码av| 欧美高清视频在线播放| 精品国产一区二区三区久久久蜜臀| 一本之道在线视频| 色偷偷一区二区三区| v片在线观看| 日韩精品第一页| 成人视屏免费看| 在线观看毛片网站|