我們知道js有幾種基本的資料類型和其他的複雜資料類型包括(對象,數組,函數),基本資料類型的賦值其實就是值的拷貝,我們稱之為值傳遞,賦值後的變量而原來的變數除了值相等之外並無其他關聯。
let x = 666 let y = x let m = 'abc' let n = m y = 888 n = 'def' console.log(x, y)//666,888 console.log(m, n)//'abc','def'
複雜資料類型的傳遞並不是這樣子的,因為將一個變數綁定到一個複雜資料類型的時候記錄的並不是這個複雜資料的值,而是存放了這個資料的一個位址訊息,當將這個變數賦值給另一個變數的時候只是將地址傳遞了過去,這兩個變數指向的其實是一個資料訊息,當改變任意一個變數的時候,另外的變數都會受到影響,這種傳遞方式我們稱之為引用傳遞
let obj1 = { a : '1', b : 2 } let obj2 = obj1 obj2.b = 3 console.log(obj1,obj2)//{a: "1", b: 3},{a: "1", b: 3}
拷貝
我們知道複雜資料型別的賦值是引用傳遞,賦值前後的變數會互相影響,在實際專案中我們常常不希望這樣子,譬如:
我們在一個view裡面有兩處用到了data(是一個Array),其一是一個list只要把data按順序展示即可,其二是一個chart需要把data逆序之後再做資料處理,這時候我們就犯難了,如果直接data.reverse()之後第一個的列表也是逆序的了,這不是我們想見的,我們需要一個方法只複製數組的值,並且這個新數組和原數組的資料位址並不一樣,這種複製方式我們稱為數組的拷貝。
let obj1 = {a:1, b:{c:2}} let shallowCopy = (src)=> { let dst = {} for (let prop in src) { if (src.hasOwnProperty(prop)) { dst[prop] = src[prop] } } return dst } let obj2 = shallowCopy(obj1) console.log(obj1,obj2) //@1 obj1.a = 6 console.log(obj2.a) //@2 obj2.b.c = 666 console.log(obj1.b.c) //@3 obj2.b = { c: 888 } console.log(obj1.b.c) //@4
上面的例子可以看出來obj1的第一層屬性是複製屬性值,沒有繼承位址的拷貝,但是第二層就是b屬性確實共享一塊記憶體位址的,這就是淺拷貝,但是在@4處obj1卻沒有收到obj2的影響,是因為屬性b是一個對象,這種引用傳遞的重新賦值,計算機會重新分配一塊新的內存來存放數據和記錄地址信息,所以這時obj1. b.c和obj2.b.c已經不是記錄的一個屬性值了
也可以理解為:拷貝是之於傳遞的,直接對複雜資料類型進行賦值是引用傳遞,不能稱之為拷貝,拷貝是原始資料的單純的資料備份,資料的記憶體位址資訊並不完全一樣,這是因為拷貝也分為淺拷貝和深拷貝。
對複雜資料類型的非嵌套的拷貝,就是只拷貝第一層的資料資訊的拷貝是淺拷貝,如果第一層的資料有複雜資料類型,則依然採用引用傳遞的方式,複製的仍然是位址訊息,透過其他方式實現的數組物件等的多層嵌套拷貝就是深拷貝。
下面我們再來看下數組和物件如何實現深淺拷貝:
數組的拷貝
-
slice方法
let arr1 = [1,2,[3,4]] let arr2 = arr1.slice(0) arr2[2].push(5) arr2.push(6) console.log(arr1,arr2)
-
concat方法
let arr1 = [1,2,[3,4]] let arr2 = arr1.concat() arr2[2].push(5) arr2.push(6) console.log(arr1,arr2)
-
for迴圈
let arr1 = [1,2,[3,4]] let arr2 = [] for(let i = 0; i<arr1.length; i++){ arr2.push(arr1[i]) } arr2[2].push(5) arr2.push(6) console.log(arr1,arr2)
-
…運算子
let arr1 = [1,2,[3,4]] let [...arr2] = arr1 arr2[2].push(5) arr2.push(6) console.log(arr1,arr2)
#以上4種數組的拷貝都是淺拷貝,要實現數組的深拷貝就要遞歸實現
let deepClone = (src)=> { let result (src instanceof Array) ? (result = []) :(result = {}) for (let key in src) { result[key] = (typeof src[key] === 'object') ? deepClone(src[key]) : src[key]//数组和对象的type都是object } return result } let arr1 = [1,2,[3,4]] let arr2 = deepClone(arr1) arr2[2].push(5) arr2.push(6) console.log(arr1,arr2)
可以發現用方面的方法arr1[2]和arr2[2]不一樣,同樣上面的深拷貝的方法也適用於物件
物件的拷貝
-
#萬能的for迴圈
let obj1 = {a:1,b:{c:2}} let obj2 = {} for(let key in obj1){ obj2[key] = obj1[key] } obj1.b.c = 6 console.log(obj1,obj2)
-
…運算符
let obj1 = {a:1,b:{c:2}} let {...obj2} = obj1 obj1.b.c = 6 console.log(obj1,obj2)
-
Object.assign()
let obj1 = {a:1,b:{c:2}} let obj2 = Object.assign({},obj1) obj1.b.c = 6 console.log(obj1,obj2)
#上面3種方法是物件的淺拷貝,再介紹2種物件的深拷貝的方法:
-
轉為字串再轉回物件
let obj1 = {a:1,b:{c:2}} let obj2 = JSON.parse(JSON.stringify(obj1)) obj1.b.c = 6 console.log(obj1,obj2)
#deepClone方法,就是上面的陣列的deepClone方法
#相關的概念
純函數
給定函數一個輸入返回一個唯一的輸出,並且不對外部環境附帶任何影響的函數我們稱為純函數,其內定義的變數在函數傳回後都會被垃圾回收機制回收。
但是如果函數的參數是數組、物件或函數時,傳入的是一個引用,對其操作會影響到原來的數據,這樣子寫的函數會產生附帶的影響,使得可讀性變低。
降低影響的方式就是對傳入參數進行深度拷貝,並賦給一個新的變量,方式原來的參數被竄改。
我們來看一個純函數的例子:
let pureFunc = (animal)=> { let newAnimal = JSON.parse(JSON.stringify(animal)) newAnimal.type = 'cat' newAnimal.name = 'Miao' return newAnimal } let wang = { type: 'dog', name: 'Wang' } let miao = pureFunc(wang) console.log(wang,miao)
透過上面的範例可以看到wang並沒有被純函數所改變值。
大家再來思考一下下面的例子,如果答對了說明你已經對本文所講的東西有了很深刻的理解了(提醒大家一下—>引用的重新賦值)
let afterChange = (obj)=>{ obj.a = 6 obj = { a: 8, b: 9 } return obj } let objIns = { a: 1, b: 2 } let objIns2 = afterChange(objIns) console.log(objIns, objIns2)
以上就是我對js的引用和傳遞的理解,如有不當之處敬請諒解,Thanks!
大家還可以看看一些其他的文章加深理解,我推薦這篇Explaining Value vs. Reference in Javascript。
相關推薦:
以上是js傳遞和拷貝詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

Node.js擅長於高效I/O,這在很大程度上要歸功於流。 流媒體匯總處理數據,避免內存過載 - 大型文件,網絡任務和實時應用程序的理想。將流與打字稿的類型安全結合起來創建POWE

Python和JavaScript在性能和效率方面的差異主要體現在:1)Python作為解釋型語言,運行速度較慢,但開發效率高,適合快速原型開發;2)JavaScript在瀏覽器中受限於單線程,但在Node.js中可利用多線程和異步I/O提升性能,兩者在實際項目中各有優勢。

JavaScript起源於1995年,由布蘭登·艾克創造,實現語言為C語言。 1.C語言為JavaScript提供了高性能和系統級編程能力。 2.JavaScript的內存管理和性能優化依賴於C語言。 3.C語言的跨平台特性幫助JavaScript在不同操作系統上高效運行。

JavaScript在瀏覽器和Node.js環境中運行,依賴JavaScript引擎解析和執行代碼。 1)解析階段生成抽象語法樹(AST);2)編譯階段將AST轉換為字節碼或機器碼;3)執行階段執行編譯後的代碼。

Python和JavaScript的未來趨勢包括:1.Python將鞏固在科學計算和AI領域的地位,2.JavaScript將推動Web技術發展,3.跨平台開發將成為熱門,4.性能優化將是重點。兩者都將繼續在各自領域擴展應用場景,並在性能上有更多突破。

Python和JavaScript在開發環境上的選擇都很重要。 1)Python的開發環境包括PyCharm、JupyterNotebook和Anaconda,適合數據科學和快速原型開發。 2)JavaScript的開發環境包括Node.js、VSCode和Webpack,適用於前端和後端開發。根據項目需求選擇合適的工具可以提高開發效率和項目成功率。

是的,JavaScript的引擎核心是用C語言編寫的。 1)C語言提供了高效性能和底層控制,適合JavaScript引擎的開發。 2)以V8引擎為例,其核心用C 編寫,結合了C的效率和麵向對象特性。 3)JavaScript引擎的工作原理包括解析、編譯和執行,C語言在這些過程中發揮關鍵作用。

JavaScript是現代網站的核心,因為它增強了網頁的交互性和動態性。 1)它允許在不刷新頁面的情況下改變內容,2)通過DOMAPI操作網頁,3)支持複雜的交互效果如動畫和拖放,4)優化性能和最佳實踐提高用戶體驗。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

MantisBT
Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

SublimeText3 英文版
推薦:為Win版本,支援程式碼提示!

SublimeText3 Linux新版
SublimeText3 Linux最新版

記事本++7.3.1
好用且免費的程式碼編輯器