真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

JavaScript中的深拷貝如何實(shí)現(xiàn)

今天小編給大家分享一下JavaScript中的深拷貝如何實(shí)現(xiàn)的相關(guān)知識點(diǎn),內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

網(wǎng)站建設(shè)哪家好,找成都創(chuàng)新互聯(lián)!專注于網(wǎng)頁設(shè)計(jì)、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序開發(fā)、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了楊浦免費(fèi)建站歡迎大家使用!

深拷貝的最終實(shí)現(xiàn)

這里先直接給出最終的代碼版本,方便想快速了解的人查看,當(dāng)然,你想一步步了解可以繼續(xù)查看文章余下的內(nèi)容:

function deepClone(target) {
    const map = new WeakMap()
    
    function isObject(target) {
        return (typeof target === 'object' && target ) || typeof target === 'function'
    }

    function clone(data) {
        if (!isObject(data)) {
            return data
        }
        if ([Date, RegExp].includes(data.constructor)) {
            return new data.constructor(data)
        }
        if (typeof data === 'function') {
            return new Function('return ' + data.toString())()
        }
        const exist = map.get(data)
        if (exist) {
            return exist
        }
        if (data instanceof Map) {
            const result = new Map()
            map.set(data, result)
            data.forEach((val, key) => {
                if (isObject(val)) {
                    result.set(key, clone(val))
                } else {
                    result.set(key, val)
                }
            })
            return result
        }
        if (data instanceof Set) {
            const result = new Set()
            map.set(data, result)
            data.forEach(val => {
                if (isObject(val)) {
                    result.add(clone(val))
                } else {
                    result.add(val)
                }
            })
            return result
        }
        const keys = Reflect.ownKeys(data)
        const allDesc = Object.getOwnPropertyDescriptors(data)
        const result = Object.create(Object.getPrototypeOf(data), allDesc)
        map.set(data, result)
        keys.forEach(key => {
            const val = data[key]
            if (isObject(val)) {
                result[key] = clone(val)
            } else {
                result[key] = val
            }
        })
        return result
    }

    return clone(target)
}

1. JavaScript數(shù)據(jù)類型的拷貝原理

先看看JS數(shù)據(jù)類型圖(除了Object,其他都是基礎(chǔ)類型):
JavaScript中的深拷貝如何實(shí)現(xiàn)
在JavaScript中,基礎(chǔ)類型值的復(fù)制是直接拷貝一份新的一模一樣的數(shù)據(jù),這兩份數(shù)據(jù)相互獨(dú)立,互不影響。而引用類型值(Object類型)的復(fù)制是傳遞對象的引用(也就是對象所在的內(nèi)存地址,即指向?qū)ο蟮闹羔槪?,相?dāng)于多個(gè)變量指向同一個(gè)對象,那么只要其中的一個(gè)變量對這個(gè)對象進(jìn)行修改,其他的變量所指向的對象也會跟著修改(因?yàn)樗鼈冎赶虻氖峭粋€(gè)對象)。如下圖:
JavaScript中的深拷貝如何實(shí)現(xiàn)

2. 深淺拷貝

深淺拷貝主要針對的是Object類型,基礎(chǔ)類型的值本身即是復(fù)制一模一樣的一份,不區(qū)分深淺拷貝。這里我們先給出測試的拷貝對象,大家可以拿這個(gè)obj對象來測試一下自己寫的深拷貝函數(shù)是否完善:

// 測試的obj對象
const obj = {
    // =========== 1.基礎(chǔ)數(shù)據(jù)類型 ===========
    num: 0, // number
    str: '', // string
    bool: true, // boolean
    unf: undefined, // undefined
    nul: null, // null
    sym: Symbol('sym'), // symbol
    bign: BigInt(1n), // bigint

    // =========== 2.Object類型 ===========
    // 普通對象
    obj: {
        name: '我是一個(gè)對象',
        id: 1
    },
    // 數(shù)組
    arr: [0, 1, 2],
    // 函數(shù)
    func: function () {
        console.log('我是一個(gè)函數(shù)')
    },
    // 日期
    date: new Date(0),
    // 正則
    reg: new RegExp('/我是一個(gè)正則/ig'),
    // Map
    map: new Map().set('mapKey', 1),
    // Set
    set: new Set().add('set'),
    // =========== 3.其他 ===========
    [Symbol('1')]: 1  // Symbol作為key
};

// 4.添加不可枚舉屬性
Object.defineProperty(obj, 'innumerable', {
    enumerable: false,
    value: '不可枚舉屬性'
});

// 5.設(shè)置原型對象
Object.setPrototypeOf(obj, {
    proto: 'proto'
})

// 6.設(shè)置loop成循環(huán)引用的屬性
obj.loop = obj

obj對象在Chrome瀏覽器中的結(jié)果:

JavaScript中的深拷貝如何實(shí)現(xiàn)

2.1 淺拷貝

淺拷貝: 創(chuàng)建一個(gè)新的對象,來接受你要重新復(fù)制或引用的對象值。如果對象屬性是基本的數(shù)據(jù)類型,復(fù)制的就是基本類型的值給新對象;但如果屬性是引用數(shù)據(jù)類型,復(fù)制的就是內(nèi)存中的地址,如果其中一個(gè)對象改變了這個(gè)內(nèi)存中的地址所指向的對象,肯定會影響到另一個(gè)對象。

首先我們看看一些淺拷貝的方法(詳細(xì)了解可點(diǎn)擊對應(yīng)方法的超鏈接):

方法使用方式注意事項(xiàng)
Object.assign()Object.assign(target, ...sources)
說明:用于將所有可枚舉屬性的值從一個(gè)或多個(gè)源對象分配到目標(biāo)對象。它將返回目標(biāo)對象。
1.不會拷貝對象的繼承屬性;
2.不會拷貝對象的不可枚舉的屬性;
3.可以拷貝 Symbol 類型的屬性。
展開語法let objClone = { ...obj };缺陷和Object.assign()差不多,但是如果屬性都是基本類型的值,使用擴(kuò)展運(yùn)算符進(jìn)行淺拷貝會更加方便。
Array.prototype.concat()拷貝數(shù)組const new_array = old_array.concat(value1[, value2[, ...[, valueN]]])淺拷貝,適用于基本類型值的數(shù)組
Array.prototype.slice()拷貝數(shù)組arr.slice([begin[, end]])淺拷貝,適用于基本類型值的數(shù)組

這里只列舉了常用的幾種方式,除此之外當(dāng)然還有其他更多的方式。注意,我們直接使用=賦值不是淺拷貝,因?yàn)樗侵苯又赶蛲粋€(gè)對象了,并沒有返回一個(gè)新對象。

手動實(shí)現(xiàn)一個(gè)淺拷貝:

function shallowClone(target) {
    if (typeof target === 'object' && target !== null) {
        const cloneTarget = Array.isArray(target) ? [] : {};
        for (let prop in target) {
            if (target.hasOwnProperty(prop)) {
                cloneTarget[prop] = target[prop];
            }
        }
        return cloneTarget;
    } else {
        return target;
    }
}


// 測試
const shallowCloneObj = shallowClone(obj)

shallowCloneObj === obj  // false,返回的是一個(gè)新對象
shallowCloneObj.arr === obj.arr  // true,對于對象類型只拷貝了引用

從上面這段代碼可以看出,利用類型判斷(查看typeof),針對引用類型的對象進(jìn)行 for 循環(huán)遍歷對象屬性賦值給目標(biāo)對象的屬性(for...in語句以任意順序遍歷一個(gè)對象的除Symbol以外的可枚舉屬性,包含原型上的屬性。查看for…in),基本就可以手工實(shí)現(xiàn)一個(gè)淺拷貝的代碼了。

2.2 深拷貝

深拷貝:創(chuàng)建一個(gè)新的對象,將一個(gè)對象從內(nèi)存中完整地拷貝出來一份給該新對象,并從堆內(nèi)存中開辟一個(gè)全新的空間存放新對象,且新對象的修改并不會改變原對象,二者實(shí)現(xiàn)真正的分離。

看看現(xiàn)存的一些深拷貝的方法:

方法1:JSON.stringify()

JSON.stringfy() 其實(shí)就是將一個(gè) JavaScript 對象或值轉(zhuǎn)換為 JSON 字符串,最后再用 JSON.parse() 的方法將JSON 字符串生成一個(gè)新的對象。(點(diǎn)這了解:JSON.stringfy()、JSON.parse())

使用如下:

function deepClone(target) {
    if (typeof target === 'object' && target !== null) {
        return JSON.parse(JSON.stringify(target));
    } else {
        return target;
    }
}

// 開頭的測試obj存在BigInt類型、循環(huán)引用,JSON.stringfy()執(zhí)行會報(bào)錯(cuò),所以除去這兩個(gè)條件進(jìn)行測試
const clonedObj = deepClone(obj)

// 測試
clonedObj === obj  // false,返回的是一個(gè)新對象
clonedObj.arr === obj.arr  // false,說明拷貝的不是引用

瀏覽器執(zhí)行結(jié)果:

JavaScript中的深拷貝如何實(shí)現(xiàn)
從以上結(jié)果我們可知JSON.stringfy() 存在以下一些問題:

  • 執(zhí)行會報(bào)錯(cuò):存在BigInt類型、循環(huán)引用。

  • 拷貝Date引用類型會變成字符串。

  • 鍵值會消失:對象的值中為FunctionUndefined、Symbol 這幾種類型,。

  • 鍵值變成空對象:對象的值中為Map、SetRegExp這幾種類型。

  • 無法拷貝:不可枚舉屬性、對象的原型鏈。

  • 補(bǔ)充:其他更詳細(xì)的內(nèi)容請查看官方文檔:JSON.stringify()

由于以上種種限制條件,JSON.stringfy() 方式僅限于深拷貝一些普通的對象,對于更復(fù)雜的數(shù)據(jù)類型,我們需要另尋他路。

方法2:遞歸基礎(chǔ)版深拷貝

手動遞歸實(shí)現(xiàn)深拷貝,我們只需要完成以下2點(diǎn)即可:

  • 對于基礎(chǔ)類型,我們只需要簡單地賦值即可(使用=)。

  • 對于引用類型,我們需要?jiǎng)?chuàng)建新的對象,并通過遍歷鍵來賦值對應(yīng)的值,這個(gè)過程中如果遇到 Object 類型還需要再次進(jìn)行遍歷。

function deepClone(target) {
    if (typeof target === 'object' && target) {
        let cloneObj = {}
        for (const key in target) { // 遍歷
            const val = target[key]
            if (typeof val === 'object' && val) {
                cloneObj[key] = deepClone(val) // 是對象就再次調(diào)用該函數(shù)遞歸
            } else {
                cloneObj[key] = val // 基本類型的話直接復(fù)制值
            }
        }
        return cloneObj
    } else {
        return target;
    }
}

// 開頭的測試obj存在循環(huán)引用,除去這個(gè)條件進(jìn)行測試
const clonedObj = deepClone(obj)

// 測試
clonedObj === obj  // false,返回的是一個(gè)新對象
clonedObj.arr === obj.arr  // false,說明拷貝的不是引用

瀏覽器執(zhí)行結(jié)果:

JavaScript中的深拷貝如何實(shí)現(xiàn)
該基礎(chǔ)版本存在許多問題:

  • 不能處理循環(huán)引用。

  • 只考慮了Object對象,而Array對象、Date對象、RegExp對象、Map對象、Set對象都變成了Object對象,且值也不正確。

  • 丟失了屬性名為Symbol類型的屬性。

  • 丟失了不可枚舉的屬性。

  • 原型上的屬性也被添加到拷貝的對象中了。

如果存在循環(huán)引用的話,以上代碼會導(dǎo)致無限遞歸,從而使得堆棧溢出。如下例子:

const a = {}
const b = {}
a.b = b
b.a = a
deepClone(a)

對象 a 的鍵 b 指向?qū)ο?b,對象 b 的鍵 a 指向?qū)ο?a,查看a對象,可以看到是無限循環(huán)的:
JavaScript中的深拷貝如何實(shí)現(xiàn)
對對象a執(zhí)行深拷貝,會出現(xiàn)死循環(huán),從而耗盡內(nèi)存,進(jìn)而報(bào)錯(cuò):堆棧溢出
JavaScript中的深拷貝如何實(shí)現(xiàn)
如何避免這種情況呢?一種簡單的方式就是把已添加的對象記錄下來,這樣下次碰到相同的對象引用時(shí),直接指向記錄中的對象即可。要實(shí)現(xiàn)這個(gè)記錄功能,我們可以借助 ES6 推出的 WeakMap 對象,該對象是一組鍵/值對的集合,其中的鍵是弱引用的。其鍵必須是對象,而值可以是任意的。(WeakMap相關(guān)見這:WeakMap)

針對以上基礎(chǔ)版深拷貝存在的缺陷,我們進(jìn)一步去完善,實(shí)現(xiàn)一個(gè)完美的深拷貝。

方法3:遞歸完美版深拷貝

對于基礎(chǔ)版深拷貝存在的問題,我們一一改進(jìn):

存在的問題改進(jìn)方案
1. 不能處理循環(huán)引用使用 WeakMap 作為一個(gè)Hash表來進(jìn)行查詢
2. 只考慮了Object對象當(dāng)參數(shù)為 Date、RegExp 、FunctionMap、Set,則直接生成一個(gè)新的實(shí)例返回
3. 屬性名為Symbol的屬性
4. 丟失了不可枚舉的屬性
針對能夠遍歷對象的不可枚舉屬性以及 Symbol 類型,我們可以使用 Reflect.ownKeys()
Reflect.ownKeys(obj)相當(dāng)于[...Object.getOwnPropertyNames(obj), ...Object.getOwnPropertySymbols(obj)]
4. 原型上的屬性Object.getOwnPropertyDescriptors()設(shè)置屬性描述對象,以及Object.create()方式繼承原型鏈

代碼實(shí)現(xiàn):

function deepClone(target) {
    // WeakMap作為記錄對象Hash表(用于防止循環(huán)引用)
    const map = new WeakMap()

    // 判斷是否為object類型的輔助函數(shù),減少重復(fù)代碼
    function isObject(target) {
        return (typeof target === 'object' && target ) || typeof target === 'function'
    }

    function clone(data) {

        // 基礎(chǔ)類型直接返回值
        if (!isObject(data)) {
            return data
        }

        // 日期或者正則對象則直接構(gòu)造一個(gè)新的對象返回
        if ([Date, RegExp].includes(data.constructor)) {
            return new data.constructor(data)
        }

        // 處理函數(shù)對象
        if (typeof data === 'function') {
            return new Function('return ' + data.toString())()
        }

        // 如果該對象已存在,則直接返回該對象
        const exist = map.get(data)
        if (exist) {
            return exist
        }

        // 處理Map對象
        if (data instanceof Map) {
            const result = new Map()
            map.set(data, result)
            data.forEach((val, key) => {
                // 注意:map中的值為object的話也得深拷貝
                if (isObject(val)) {
                    result.set(key, clone(val))
                } else {
                    result.set(key, val)
                }
            })
            return result
        }

        // 處理Set對象
        if (data instanceof Set) {
            const result = new Set()
            map.set(data, result)
            data.forEach(val => {
                // 注意:set中的值為object的話也得深拷貝
                if (isObject(val)) {
                    result.add(clone(val))
                } else {
                    result.add(val)
                }
            })
            return result
        }

        // 收集鍵名(考慮了以Symbol作為key以及不可枚舉的屬性)
        const keys = Reflect.ownKeys(data)
        // 利用 Object 的 getOwnPropertyDescriptors 方法可以獲得對象的所有屬性以及對應(yīng)的屬性描述
        const allDesc = Object.getOwnPropertyDescriptors(data)
        // 結(jié)合 Object 的 create 方法創(chuàng)建一個(gè)新對象,并繼承傳入原對象的原型鏈, 這里得到的result是對data的淺拷貝
        const result = Object.create(Object.getPrototypeOf(data), allDesc)

        // 新對象加入到map中,進(jìn)行記錄
        map.set(data, result)

        // Object.create()是淺拷貝,所以要判斷并遞歸執(zhí)行深拷貝
        keys.forEach(key => {
            const val = data[key]
            if (isObject(val)) {
                // 屬性值為 對象類型 或 函數(shù)對象 的話也需要進(jìn)行深拷貝
                result[key] = clone(val)
            } else {
                result[key] = val
            }
        })
        return result
    }

    return clone(target)
}



// 測試
const clonedObj = deepClone(obj)
clonedObj === obj  // false,返回的是一個(gè)新對象
clonedObj.arr === obj.arr  // false,說明拷貝的不是引用
clonedObj.func === obj.func  // false,說明function也復(fù)制了一份
clonedObj.proto  // proto,可以取到原型的屬性

在遍歷 Object 類型數(shù)據(jù)時(shí),我們需要把 Symbol 類型的鍵名也考慮進(jìn)來,所以不能通過 Object.keys 獲取鍵名或 for...in 方式遍歷,而是通過Reflect.ownKeys()獲取所有自身的鍵名(getOwnPropertyNamesgetOwnPropertySymbols 函數(shù)將鍵名組合成數(shù)組也行:[...Object.getOwnPropertyNames(obj), ...Object.getOwnPropertySymbols(obj)]),然后再遍歷遞歸,最終實(shí)現(xiàn)拷貝。

瀏覽器執(zhí)行結(jié)果:
JavaScript中的深拷貝如何實(shí)現(xiàn)
可以發(fā)現(xiàn)我們的cloneObj對象和原來的obj對象一模一樣,并且修改cloneObj對象的各個(gè)屬性都不會對obj對象造成影響。其他的大家再多嘗試體會哦!

以上就是“JavaScript中的深拷貝如何實(shí)現(xiàn)”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習(xí)更多的知識,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


標(biāo)題名稱:JavaScript中的深拷貝如何實(shí)現(xiàn)
本文網(wǎng)址:http://weahome.cn/article/ihoegd.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部