摘要:正文討論深淺拷貝,首先要從的基本數(shù)據(jù)類型說起根據(jù)中的變量類型傳遞方式,分為值類型和引用類型,值類型變量包括。當(dāng)你拷貝的對象有多級的時候,就是深拷貝。數(shù)據(jù)不存在則對其拷貝。
前言:
本文主要閱讀對象:對深淺拷貝印象模糊對初級前端,想對js深淺拷貝聊一聊的中級前端。 如果是對這些有完整對認(rèn)知體系和解決方法的大佬,可以選擇略過。
正文:
討論深淺拷貝,首先要從js的基本數(shù)據(jù)類型說起:
根據(jù) JavaScript 中的變量類型傳遞方式,分為值類型和引用類型,
值類型變量包括 Boolean、String、Number、Undefined、Null。
引用類型包括了 Object 類的所有, 如 Date、Array、Function 等。在參數(shù)傳遞方式上,值類型是按值傳遞,引用類型是按地址傳遞。
我們來看一下這兩則有什么區(qū)別:
//eg1: // 值類型 var a = 10 var b = a b = 20 console.log(a) // 10 console.log(b) // 20 //解析:上述代碼中,a b都是值類型,兩者分別修改賦值,相互之間沒有任何影響。再看引用類型的例子: //eg2 // 引用類型 var a = {x: 10, y: 20} var b = a b.x = 100 b.y = 200 console.log(a) // {x: 100, y: 200} console.log(b) // {x: 100, y: 200}
解析: 上述代碼中,a b都是引用類型。在執(zhí)行了b = a之后,修改b的屬性值,a的也跟著變化。因?yàn)閍和b都是引用類型,指向了同一個內(nèi)存地址,即兩者引用的是同一個值,因此b修改屬性時,a的值隨之改動。
**那到底什么是深淺拷貝呢?**
解析:深淺拷貝是拷貝對象的`深度`來說的:
當(dāng)你想拷貝的對象只有一級時,就是淺拷貝。
當(dāng)你拷貝的對象有多級的時候,就是深拷貝。
再回到上面的那個例子: 在例子能看出來,如果直接采用“=”賦值,這種類型,當(dāng)我們改變b的值時候,a也會隨之改變的,假如不允許改變a呢? 這時,深拷貝就出場了。
function clone(val){ if(!val && typeof val !== "object"){ return } const newArr = toString.call(val) === ["object Array"] ? [] : {} for (let item in val) { if(typeof val[item] === "object") { newArr[item] = clone(item) }else { newArr[item] = val[item] } } return newArr } //測試: var a = {x: 10, y: 20} var b = clone(a) b.x = 100 b.y = 200 console.log(a) // {x: 10, y: 20} console.log(b) //{x: 100, y: 200}
解析:
對于這個深拷貝,大家看到這里應(yīng)該都可以看明白:主要思路就是淺拷貝 + 遞歸。
這有幾個點(diǎn)需要注意:
1. 判斷接收到的參數(shù)類型。
2. 使用toString.call()判斷更加嚴(yán)謹(jǐn)。
3. 考慮對其他引用類型的數(shù)據(jù)兼容(這里并沒有對每種類型的數(shù)據(jù)組做判斷,引用類型包括了 Object 類的所有,如 Date、Array、Function 等。)
需要注意的是:用遞歸的話,會有一下幾個需要特別注意的點(diǎn): 1. 當(dāng)數(shù)據(jù)層次很深時,容易爆棧(棧溢出) 解決辦法=> a. 消除尾遞歸 b. 改用循環(huán) 2. “循環(huán)引用”,會導(dǎo)致死循環(huán) 解決辦法 => a. 循環(huán)檢測 b. 暴力破解 第一種:數(shù)據(jù)廣度特別大,層次特別深 第二種:類似下面的這種情況 var a = {} a.a = a clone(a) //死循環(huán) ES6中一行代碼實(shí)現(xiàn)深拷貝:JSON.parse(JSON.stringify(source)) 注釋:JSON.stringify()其實(shí)利用了“循環(huán)檢測”機(jī)制
這里給大家推薦一篇循環(huán)解決遞歸的方法: 可以保持拷貝數(shù)據(jù)以后的引用關(guān)系
function cloneForce(x) { const uniqueList = []; // 用來去重 let root = {}; // 循環(huán)數(shù)組 const loopList = [ { parent: root, key: undefined, data: x, } ]; while(loopList.length) { // 深度優(yōu)先 const node = loopList.pop(); const parent = node.parent; const key = node.key; const data = node.data; // 初始化賦值目標(biāo),key為undefined則拷貝到父元素,否則拷貝到子元素 let res = parent; if (typeof key !== "undefined") { res = parent[key] = {}; } // 數(shù)據(jù)已經(jīng)存在 let uniqueData = find(uniqueList, data); if (uniqueData) { parent[key] = uniqueData.target; continue; // 中斷本次循環(huán) } // 數(shù)據(jù)不存在 // 保存源數(shù)據(jù),在拷貝數(shù)據(jù)中對應(yīng)的引用 uniqueList.push({ source: data, target: res, }); for(let k in data) { if (data.hasOwnProperty(k)) { if (typeof data[k] === "object") { // 下一次循環(huán) loopList.push({ parent: res, key: k, data: data[k], }); } else { res[k] = data[k]; } } } } return root; } function find(arr, item) { for(let i = 0; i < arr.length; i++) { if (arr[i].source === item) { return arr[i]; } } return null; } //eg1: var a = { a1: b, a2: b, } var b = {}; a.a1 === a.a2 // true var c = cloneForce(a); c.a1 === c.a2 // true 引用保持一致 //eg2: var a = {}; a.a = a; console.log(cloneForce(a))//還可以破解循環(huán)引用
主要思路是:
聲明一個數(shù)組對象(父對象,key,value),其有值時,取其最后一個對象;
判斷key有值,代表當(dāng)前級下有子級,則拷貝到子元素。如果數(shù)據(jù)已經(jīng)存在,則中斷本次循環(huán)。數(shù)據(jù)不存在則對其拷貝。
有一點(diǎn)需要注意:cloneForce在對象數(shù)量很多時會出現(xiàn)很大的問題,如果數(shù)據(jù)量很大不適合使用cloneForce。
有興趣的同學(xué),可以前往 深拷貝的終極探索(90%的人都不知道),查看更多的細(xì)節(jié)。
總結(jié):
如果覺得對你有幫助,請給作者一點(diǎn)小小的鼓勵,
點(diǎn)個贊或者收藏吧。 有需要溝通的請聯(lián)系我:
微信( wx9456d ) 或
郵箱( allan_liu986@163.com )
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/6669.html
摘要:一篇文章徹底說清的深拷貝淺拷貝這篇文章的受眾第一類業(yè)務(wù)需要急需知道如何深拷貝對象的開發(fā)者。這篇文章分享的目的更多還是希望用一篇文章整理清楚深淺拷貝的含義遞歸實(shí)現(xiàn)思路以及小伙伴們?nèi)绻褂昧诉@種黑科技一定要清楚這樣寫的優(yōu)缺點(diǎn)。 一篇文章徹底說清JS的深拷貝and淺拷貝 這篇文章的受眾 第一類,業(yè)務(wù)需要,急需知道如何深拷貝JS對象的開發(fā)者。 第二類,希望扎實(shí)JS基礎(chǔ),將來好去面試官前秀操作...
摘要:一篇文章徹底說清的深拷貝淺拷貝這篇文章的受眾第一類業(yè)務(wù)需要急需知道如何深拷貝對象的開發(fā)者。這篇文章分享的目的更多還是希望用一篇文章整理清楚深淺拷貝的含義遞歸實(shí)現(xiàn)思路以及小伙伴們?nèi)绻褂昧诉@種黑科技一定要清楚這樣寫的優(yōu)缺點(diǎn)。 一篇文章徹底說清JS的深拷貝and淺拷貝 這篇文章的受眾 第一類,業(yè)務(wù)需要,急需知道如何深拷貝JS對象的開發(fā)者。 第二類,希望扎實(shí)JS基礎(chǔ),將來好去面試官前秀操作...
摘要:一篇文章徹底說清的深拷貝淺拷貝這篇文章的受眾第一類業(yè)務(wù)需要急需知道如何深拷貝對象的開發(fā)者。這篇文章分享的目的更多還是希望用一篇文章整理清楚深淺拷貝的含義遞歸實(shí)現(xiàn)思路以及小伙伴們?nèi)绻褂昧诉@種黑科技一定要清楚這樣寫的優(yōu)缺點(diǎn)。 一篇文章徹底說清JS的深拷貝and淺拷貝 這篇文章的受眾 第一類,業(yè)務(wù)需要,急需知道如何深拷貝JS對象的開發(fā)者。 第二類,希望扎實(shí)JS基礎(chǔ),將來好去面試官前秀操作...
摘要:深拷貝相比于淺拷貝速度較慢并且花銷較大。所以在賦值完成后,在棧內(nèi)存就有兩個指針指向堆內(nèi)存同一個數(shù)據(jù)。結(jié)果如下擴(kuò)展運(yùn)算符只能對一層進(jìn)行深拷貝如果拷貝的層數(shù)超過了一層的話,那么就會進(jìn)行淺拷貝那么我們可以看到和展開原算符對于深淺拷貝的結(jié)果是一樣。 JS中數(shù)據(jù)類型 基本數(shù)據(jù)類型: undefined、null、Boolean、Number、String和Symbol(ES6) 引用數(shù)據(jù)類型:...
摘要:理解的函數(shù)基礎(chǔ)要搞好深入淺出原型使用原型模型,雖然這經(jīng)常被當(dāng)作缺點(diǎn)提及,但是只要善于運(yùn)用,其實(shí)基于原型的繼承模型比傳統(tǒng)的類繼承還要強(qiáng)大。中文指南基本操作指南二繼續(xù)熟悉的幾對方法,包括,,。商業(yè)轉(zhuǎn)載請聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請注明出處。 怎樣使用 this 因?yàn)楸救藢儆趥吻岸耍虼宋闹兄豢炊?8 成左右,希望能夠給大家?guī)韼椭?...(據(jù)說是阿里的前端妹子寫的) this 的值到底...
閱讀 2108·2021-11-11 16:55
閱讀 3178·2021-10-11 10:58
閱讀 3052·2021-09-13 10:28
閱讀 3982·2021-07-26 23:57
閱讀 1031·2019-08-30 15:56
閱讀 1339·2019-08-29 13:15
閱讀 1275·2019-08-26 18:18
閱讀 1278·2019-08-26 13:44